JVM的主要组成部分及其作用
概括的讲包含两个子系统和两个组件,分别是类加载子系统、执行引擎及运行时数据区、本地接口
类加载:根据给定的全限定类名装载class文件到运行时数据区的方法区中
执行引擎:执行class中的指令
本地接口:与native library交互,是与其他编程语言交互的接口
运行时数据区:即JVM内存
首先编译器把Java代码转换成字节码,类加载器再把字节码加载到内存中,将其放到运行时数据区的方法区内;字节码文件是JVM的一套指令集规范,不能直接交给底层操作系统去执行,因此需要特定的命令解析执行引擎,将字节码翻译成底层系统指令,再由CPU去执行,这个过程中需要调用其他语言的本地接口来实现整个程序的功能。
类的加载指的是将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个java.lang.Class对象,用来封装类在方法区内的数据结构。
JVM运行时数据区
Java虚拟机在执行Java程序的过程中会把它所管理的内存区域划分为若干个不同的数据区域。不同虚拟机的运行时数据区可能略有不同,但都遵循Java虚拟机规范,Java虚拟机规范规定的区域分为以下5个部分:
程序计数器:当前线程所执行的字节码的行号指示器,字节码解析器的工作是通过改变这个计数器的值,来选取下一条需要执行的字节码指令,分支、循环、跳转、异常处理、线程恢复等基础功能都需要依赖此;
Java虚拟机栈,用于存放局部变量表、操作数栈、动态链接、方法出口等信息;
Java堆:Java虚拟机中内存最大的一块,被所有线程所共享,几乎所有的对象实例都在这里分配空间;
方法区:用于存储已被虚拟机加载的类的信息、常量、静态变量、即时编译后的代码等数据;
本地方法栈:与虚拟机栈的作用一样,只不过虚拟机栈是服务Java方法的,而本地方法栈是为虚拟机调用Native方法服务的;
深拷贝和浅拷贝
浅拷贝:只是增加了一个指针指向已存在的内存地址,
深拷贝:不仅增加了一个指针并且同时申请了一个新的内存,使这个增加的指针指向这个新的内存。
使用深拷贝,不会出现浅拷贝时释放同一个内存的错误。
堆栈的区别
物理地址:堆的物理地址分配是不连续的,因此性能上慢些,所以GC的时候,有各种算法,如标记-清除,复制,标记-压缩,分代;栈使用的数据结构中的栈,先进后出的原则,物理地址是连续的,所以性能快。
内存分别:堆因为是不连续的,所以分配的内存是在运行期确认的,因此大小不固定,一般堆大小远远大于栈;栈是连续的,所以分配的内存大小要在编译期就确认,大小是固定的。
存放内容:堆存放的是对象的实例和数组,因此更加关注数据的存储;栈存放的是局部变量、操作数栈、返回结果,更加关注的是程序方法的执行。
静态变量放在方法区,静态对象还是放在堆
堆对于整个应用程序都是共享、可见的;栈只对于线程使可见的,是线程私有的,生命周期和线程相同。
队列和栈的区别
队列:先进先出(FIFO)
栈:先进后出(LIFO)