当前位置: 首页 > news >正文

Java100道面试题

1.JVM内存结构

VM内存结构指的是JVM运行时数据区结构,它主要包含以下几个部分:

堆(Heap):线程共享。

JVM堆(Heap)是Java虚拟机中的一块内存区域(所有线程共享),主要用于存储对象实例和数组。堆被划分为三个部分:年轻代、老年代和永久代(在JDK8中取消了永久代),其中,年轻代又被划分为Eden区、Survivor区(含:S0和S1)

方法区(Method Area):线程共享。

在JDK8及之前,方法区属于永久代,而在JDK8之后,永久代被移除,方法区被移到了本地内存中,即:元空间(Meta Space)。

元空间逻辑上属于堆的一部分,但是为了与堆进行区分,通常又叫非堆。元空间是各个线程共享的内存区域,它主要存储两部分内容:

虚拟机栈(VM Stack):线程私有。

虚拟机栈为线程私有,它描述的是Java方法执行的内存模型。每个栈由多个栈帧(Stack Frame)组成,每个方法被执行时,Java虚拟机都会同步创建一个栈帧用于存储该方法的局部变量表、操作数栈、动态链接、方法返回地址等信息

程序计数器(Program Counter Register):线程私有

  • 每个线程都有一个独立的程序计数器,各线程之间计数器互不影响,独立存储。

  • 执行Java方法时,程序计数器的值不为空;执行Native本地方法时,程序计数器的值为空(Undefined)。

  • 程序计数器是唯一一个在Java虚拟机规范中没有规定任何内存溢出情况的区域

本地方法栈(Native Method Stack):线程私有

本地方法栈与虚拟机栈类似,不同的是虚拟机栈为虚拟机执行Java方法服务,而本地方法栈则为虚拟机使用到的Native方法服务。在Java程序调用Native方法时,Native方法所需要的内存空间在本地方法栈中开辟。

2.垃圾回收机制(GC)

JVM的垃圾回收机制:GC,是Java提供的对于内存自动回收的机制。

GC(Garbage Collection)是Java虚拟机(JVM)中的一项重要功能,用于自动管理堆内存中不再使用的对象,释放其占用的内存空间。GC通过标记和回收无效对象来实现内存的回收和释放,以避免内存泄漏和溢出。

        下面是GC的主要工作原理和过程:

        1、对象的标记:GC首先标记出所有活动对象,即仍然被引用或可达的对象。它从一组根对象开始,逐步遍历对象图,将可达的对象标记为活动对象,未标记的对象则被认为是无效的。

        2、垃圾回收:在标记完成后,GC会对未标记的对象进行回收。具体的回收算法可以是不同的,常见的算法包括标记-复制算法、三色标记算法等。

        3、内存压缩和整理:某些垃圾回收算法在回收完对象后,可能会产生内存碎片。为了优化内存使用,GC可能会进行内存压缩和整理操作,使得分配的对象在内存中连续存放,减少内存碎片的影响。

        GC的优点是可以自动管理内存,减少了手动内存管理的复杂性,避免了内存泄漏和溢出的问题。但是,GC也会带来一定的性能开销。因此,在开发Java应用程序时,需要合理配置GC的参数和调整垃圾回收策略,以平衡性能和内存的使用。
 

3.JVM调优的方法?

1. 明确优化目标

在开始调优之前,首先需要明确优化的目标。这包括响应时间、吞吐量、内存使用率等指标。明确目标后,可以更有针对性地进行调优

2. 监控和分析
  1. 使用监控工具:利用Java VisualVM、JConsole、JProfiler等监控工具,分析CPU使用率、内存使用情况和垃圾回收频率等指标。
  2. 分析GC日志:通过GC日志,了解垃圾回收的频率、停顿时间以及内存使用情况,帮助定位性能瓶颈。

3. 确定调优参数

根据监控和分析的结果,确定需要调整的JVM参数。常见的JVM参数包括:

  1. 堆内存设置:通过-Xms和-Xmx设置初始和最大堆内存大小。
  2. 垃圾收集器选择:选择合适的垃圾收集器,如G1、CMS等,以优化内存管理
  3. 线程栈大小:通过-Xss设置每个线程的堆栈大小,以减少线程上下文切换的开销。

4. 实施调优策略

  1. 调整JVM参数:根据确定的调优参数,调整JVM的启动参数。
  2. 优化代码:减少不必要的对象创建、使用高效的数据结构、避免过度的同步等,都可以显著提升应用的性能。

5. 持续观察和调整

  1. 持续监控:在调整参数和优化代码后,使用监控工具持续观察应用的性能变化。
  2. 对比性能指标:通过对比调优前后的性能指标,确认是否达到了预期的优化目标。
  3. 重新评估和调整:如果没有达到目标,可能需要重新评估调优策略,进行进一步的调整。

6. 定期评估和优化

性能调优是一个持续的过程。随着应用的演变和用户负载的变化,可能需要定期重新评估和调整JVM参数。定期的性能测试和监控可以帮助及时发现潜在问题,确保系统的稳定性和高效性。

4.堆和栈的区别?

Java堆和栈是Java虚拟机(JVM)中的两个重要概念,它们在内存管理、存储对象和执行线程等方面有明显的区别。

1、内存分配和管理
Java堆是动态分配的内存区域,主要用来存储对象实例。在Java中,对象是通过堆内存进行分配的。当创建一个对象时,Java虚拟机会在堆上分配相应的内存空间,并自动进行垃圾回收和内存管理。堆的大小可以在运行时动态调整,通过JVM参数进行配置。
相比之下,Java栈是线程私有的,每个线程在创建时都会创建一个栈。栈由一系列栈帧组成,每个栈帧对应一个方法调用。栈主要用于存储基本数据类型、对象引用和方法的局部变量。每个方法从调用开始到执行结束的过程,对应一个栈帧在栈内存中的入栈到出栈的过程。

2、存储内容
Java堆主要用于存储对象实例,它是所有线程共享的一块内存区域。堆中可以存储任意类型的对象,包括数组和类的实例。堆中的对象可以是任意的Java对象,如String、Integer等。堆是由垃圾回收器自动管理的,当一个对象不再被引用时,垃圾回收器会自动回收该对象占用的堆内存。
Java栈主要存储基本数据类型、对象引用和方法的局部变量。每个方法从调用开始到执行结束的过程,对应一个栈帧在栈内存中的入栈到出栈的过程。每个方法从调用直至执行完成的过程,对应着一个栈帧在虚拟机栈中入栈到出栈的过程。栈帧是用于支持虚拟机进行方法执行的数据结构,也是虚拟机运行时数据区中的一块内存区域。

3、线程执行和生命周期
Java堆是所有线程共享的内存区域,而Java栈是线程私有的。每个线程在创建时都会创建一个自己的栈,并且这个栈的生命周期与线程相同。当线程启动时,它的栈随之创建;当线程结束时,它的栈也随之销毁。每个方法的执行都伴随着一个栈帧的入栈和出栈过程,方法的执行过程对应着其栈帧在栈内存中的生命周期。
相比之下,堆的生命周期与应用程序的启动和结束相同。当应用程序启动时,堆被创建;当应用程序结束时,堆随之销毁。垃圾回收器自动管理堆内存的回收和释放。

4、性能影响
由于Java堆是所有线程共享的内存区域,因此对堆的操作是共享的。这意味着多个线程可以同时访问堆中的数据,并进行读写操作。这种共享性使得堆在某些情况下可以提高多线程程序的性能。
相比之下,Java栈是线程私有的,每个线程都有自己的独立栈。因此,对栈的操作是线程独占的。这意味着同一时间只有一个线程可以访问某个栈帧中的数据,这有助于避免多线程并发访问带来的问题。但是,由于每个线程都有自己的独立栈,因此可能会占用更多的内存空间

5.JVM使用命令

jvm 的一些命令_jvm命令-CSDN博客

6.class.forName和Classload的区别

1.相同点
两者都可以对类进行加载。
对于任意一个类/对象,我们都能够通过反射能够调用这个类的所有属性方法。

2.不同点
抽象类ClassLoader中实现的方法loadClass,loadClass只是加载,不会解析更不会初始化所反射的类。常用于做懒加载,提高加载速度,使用的时候再通过.newInstance()真正去初始化类。 

7.谈谈JDK1.8特性有哪些

Lambda表达式 类似于ES6中的箭头函数
接口的默认方法和静态方法
新增方法引用格式
新增Stream类
新的日期API,Datetime,更方便对日期的操作
引入Optional,在SpringData中使用较多,然后再通过get获取值,主要用于防止NPE。
支持Base64
注解相关的改变
支持并行(parallel)数组
对并发类(Concurrency)的扩展。
JavaFX。

JDK1.8常用新特性_jdk1.8的新特性-CSDN博客

8.反射获取类中的所有方法和获取类中的所有属性

9.懒汉和饿汉模式的区别?口述两种模式。 

在饿汉式单例模式中,“饿” 体现的是一种急切的状态。就好像一个很饿的人,在看到食物(这里类比于单例对象)的时候,会迫不及待地先把食物拿到手(创建单例对象)。在这个模式下,单例对象在类加载阶段就被创建出来,而不是等到真正需要使用这个对象的时候才去创建。这种方式比较急切,所以被称为 “饿汉模式”。

在懒汉模式下,实例在第一次使用时才进行创建,因此称为“懒汉”,在需要被用的时候被创建,突出一个字“懒”

10.如何保证单例模式在多线程中的线程安全

私有构造函数:通过将构造函数设置为私有,我们禁止了外部类直接实例化Singleton类。这确保了只能通过getInstance()方法来获取实例。

volatile关键字:volatile关键字确保了instance变量的可见性。当一个线程修改了一个volatile变量的值时,其他线程能够立即看到这个变化。这样可以避免线程之间的缓存不一致问题。

双重检查锁定(Double-Checked Locking):为了减少同步开销,我们在第一次检查instance是否为空后,才进入同步块。这是因为大多数情况下,实例已经被创建,不需要进入同步块。只有在第一次创建实例时才会需要同步。

同步块:在同步块内部,我们再次检查instance是否为空,以确保只有一个线程能够创建实例。这是为了防止多个线程同时通过了第一个空检查并尝试创建多个实例。

11.spring运用了什么设计模式?讲一下哪些部分运用到了这些设计模式

12.说说你都知道哪些设计模式?最常用的有哪些?总共有几种设计模式?

13.redis可以储存哪几种数据类型?你的项目里都用redis存储哪些数据

14.redis有哪些常用操作?(命令)

15.redis缓存和数据库怎么保持一致?

16.redis可以持久化么?如何持久化?

17.redis分布式锁怎么使用?

18.redis缓存数据丢失怎么使用?

19.redis如果崩溃了如何快速恢复?

20.redis是如何部署的?是单个部署还是集群部署?为什么这么做?

21.什么是缓存穿透、缓存击穿、缓存雪崩?如何解决?

22.List和set和map的区别?

23.arrarylist,linkedlist;arraylist内部扩容机制是怎样的?

24.hashmap扩容机制,HashMap的底层原理

25.hashmap的底层用什么储存的?线程是否安全?

26.final和finally和finalize的区别

27.String、StringgBuilder、StringBuffer的区别

28.重写equals已经能比较两个对象了,为什么还要重写hashcode方法

29.基本数据和包装数据类型的区别?

30.社么是多态?举个例子

31.抽象类和接口的区别

32.java中的几种基本类型,各占用多少字节?

33.运行时异常有哪些?

34.对面向对象的理解?面向对象有的特征有哪些?
35.HTTP报文有哪几部分组成?Get/Post有什么区别

36.转发(forward)和重定向(redirect)的区别

37.什么是AOP?什么是IOC/

38.AOP用到什么设计模式

39.SpringBean的生命周期

40.SpringIOC是怎么实现了依赖注入,有几种依赖注入方式?

41.SpringMVC的执行流程

42.SpringMVC与springboot的区别?

43.SpringMVC的事物有过哪几种?怎么处理的?

44.SpringBoot在初始化以前执行一个方法,应该怎么做?初始化以后在执行一个方法应该怎么做,有哪几种方法去做

45.在项目中会经常用到哪些注解?讲出十个

46.@Autowired注入和自己new一个对象有什么区别?

47.myBaties防sql注入的操作

48.Mybaties都要用到哪些标签

49.#和$的区别

50.Mybaties如何实现循环?

51.@Component,@Controller,@Repository,@Service有何区别?

52.SpringBoot中的注解?SpringBoot的核心注解是什么?

53.springBoot要发布一个接口需要几个注解?

54.SpringBoot的运行原理/启动流程?

55.SpringBoot项目启动类上有什么注解?除了常用的还有什么注解?

56.SpringBoot配置文件怎么配?

57.SpringBoot怎么整合MyBatis

58.SpringBoot项目启动时配置文件加载顺序

59.事物的隔离级别

60.事物的七种传播方式

61.过滤器和拦截器的区别?

62.Spring运用了什么设计模式?讲一下哪些部分用到了这些设计模式

63.去重SQL语句怎么写

64.MYSQL大批量数据如何导入

65.mysql如何做大数据量分页

66.mysql索引类型

67.怎么避免索引失效

68.了解过MySQL数据库储存引擎么?

69.讲讲mysql最左匹配原则

70.笛卡尔积是什么

71.脏读、幻读、不可重复读

72.数据库的优化方案?一个SQL语句如何让他搜索变快?你们遇到的数据量最大的表是什么样,有多少条数据。

73.慢日志怎么使用?MYSQL怎么发现动作速度慢的SQL语句?

74.分库分表是怎么做的?数据库分表分库后怎么保证其一致性?

75.MySQL表为何产生死锁?如何避免死锁?

76.oracle与mysql的区别?mysql分页和oracle分页的区别?

77.Mysql默认的事务隔离级别?

78.Mysql的索引的数据结构?

79.Mysql中count()和count(1)区别

80.Sql看执行计划?

81.创建线程的方式及其区别

82.mysq1锁的粒度?并发编程锁到行还是表?(介绍页级、表级、行级)

83.什么是线程和进程?它们之间的区别是什么?

84.请解释一下线程的几种状态以及状态之间的转换关系

85.什么是线程上下文切换?

86.项目中是否用到过多线程?怎么用的?

87.Sleep()和wait()区别?Start()和run()区别?wait,sleep, notify的区别?

88.线程安全指的是什么?如何保障线程安全

89.线程锁有几种方式?使用的关键字?

90.什么是乐观锁和悲观锁?

91.synchronized和]ock的区别。Synchronize关键字修饰方法和修饰代码块的区别

92.如何保证单例模式在多线程中的线程安全性

93.线程池的常见类型石那些?

94.线程池中线程的停止?

95.线程池中的线程数多少合适?

96.线程池一般是自动创建的还是手动?为什么?

97.ReentrantLock和synchronized的别是什么

98.什么是死锁?如何避免死锁?

99.Java 中线程间通信有哪些方式


http://www.mrgr.cn/news/82663.html

相关文章:

  • 【C语言程序设计——循环程序设计】统计海军鸣放礼炮声数量(头歌实践教学平台习题)【合集】
  • opencv摄像头标定程序实现
  • DINOv2+Qwen2.5-VL-2B+LoRA实现image caption的微调
  • 【微服务】4、服务保护
  • 小程序26-事件绑定和事件对象
  • 小白学Pytorch
  • 网站常用功能模块-鉴权
  • 【Rust自学】10.6. 生命周期 Pt.2:生命周期的语法与例子
  • 后台管理系统用户退出登录方案实现
  • 4进货+后台事务
  • Kubernetes Gateway API-3-TLS配置
  • 如何在 Hive SQL 中处理复杂的数据类型?
  • 安卓触摸对焦
  • 如何操作github,gitee,gitcode三个git平台建立镜像仓库机制,这样便于维护项目只需要维护一个平台仓库地址的即可-优雅草央千澈
  • 如何使用 Ansys OptiSlang 同时运行多个参数化设计研究
  • 当今世界如何减少暴戾之气和矛盾纷争
  • 【Rust自学】10.5. 生命周期 Pt.1:生命周期的定义与意义、借用检查器与泛型生命周期
  • Linux 基础七 内存
  • 修改secure-file-priv参数-mysql5.7.26限制不允许导入或导出的解决方法
  • GNU链接器简介-2
  • Ubuntu 下载安装 Consul1.17.1
  • Branch-Solve-Merge Improves Large Language Model Evaluation and Generation
  • 高中数学部分基础知识
  • 【51单片机零基础-chapter4:LED数码管】
  • 【51单片机零基础-chapter2:灯独立点亮,自定义点亮,跑马灯点亮,函数】
  • 1.1.3 插入排序