1.方法区
方法区是一块大小可调节,线程共享的区域,在jvm启动时就被创建,关闭jvm就会释放这一区域。方法区的大小决定了系统可以保存多少个类,如果系统定义了太多的类,导致方法区溢出, 方法区用来存储加载类信息,以及编译期编译后的信息。方法区还包含了一个特殊的区域“运行时常量池”。
方法区,堆,栈交互关系 :
方法区存储类信息(元信息)
堆中存储创建的对象
栈中存储对象引用
方法区的内部结构:
方法区用于存储被虚拟机加载的类信息,常量,静态变量,运行时常量池等。
运行时常量池就是一张表,虚拟机会根据这张表,找到要执行的类名,方法名,参数类型,字面量(常量)等信息。
运行时常量池存放编译期间生成的各种字面量和符号引用
方法区的垃圾回收:
方法区只有在 full GC(整堆收集) 的时候才会进行垃圾回收,回收频率不是很高
主要回收类信息,但是类一旦被加载 要回收的条件比较苛刻
必须满足以下几点类才可被回收:
1.在堆中,该类及其子类的对象都不存在了
2.该类的类加载器不存在了
3.该类的Class对象不存在了
也可以认为类一旦被加载就不会被卸载了.
2.本地方法接口:
用native关键字修饰的方法称为一个本地方法, 该方法没有方法体. 底层并非用java语言实现
常用的本地方法有:
计算哈希码时调用的hashcode();
启动线程时start()方法底层调用的star0();
IO读取本地文件read()方法 底层调用的read0()
为什么用本地方法 :
java语言需要与外部的环境进行交互(例如需要访问内存,硬盘,其他的硬件设备),直接访问操作系统提供的接口即可.
3.执行引擎
作用: 将加载到内存中的字节码(不是直接运行的机器码), 解释/编译为不同平台的机器码.
代码执行的过程:
.java将代码 ---编译-->.class文件(字节码) 在开发期间,由jdk提供的编译器(javac)进行源码编译 (前端编译)
.class文件(字节码)----解释/编译---> 机器码 (后端编译,在运行时,由执行引擎完成的)
什么是解释器?什么是 JIT 编译器
解释器: 将字节码逐行解释执行, 效率低
编译器(JIT just in time 即时编译器): 将字节码编译,缓存起来,执行更高效。
编译器会将一些频繁执行的热点代码进行编译,并缓存到方法区中,以后执行效率提高了.
程序启动后,先使用解释器立即执行,省去了编译时间,
程序运行一段时间后,对热点编译缓存,提高后续执行效率
所以采用的解释器和编译器结合的方案.