关于类的加载过程,和jvm调优
一个class类的大小可以用jol-core jar包来查看它的大小,一个类的大小基本上由请求头,实例对象,对其填充组成的,对其填充里面又有mark word,类型指针,数组长度组成,mark word 8B,里面存放多线程的锁对象,类型指针,指向线性地址,开启指针压缩是4b,未开启指针压缩是8B,实例对象,就是他的成员对象,pravite 里面装对象, 然后就是对齐填充,都是以8B对齐的。注意未开启指针压缩,会在数组长度之后开一个对齐填充。
类的加载过程就是将class文件加载进虚拟机的过程,会有一个验证,验证class文件是否正确一般class文件16进制都是以魔术开头,然后准备,就是给类的静态变量分配内存空间,赋初值元空间(?),class对象也是放元空间,对象是放在堆里面的,下一步就是解析,解析就是将符号引用替换成直接引用(比如将static 的main方法的符号变成直接指向内存地址的指针或句柄等 ),这就是所谓静态链接过程,是在类加载中完成,动态链接是指程序运行期间的符号引用变成直接引用,最后一步就是初始化,给初值的初始化制定的值类的静态变量初始化,执行静态代码块
加载,类的加载都是用到时候在加载,比如说有一个pravite成员对象,会在执行main方法中的那行new对象的方法才会执行。
new对象的过程会先去加载父类,然后自己的属性和方法,可以用jclasslib插件查看加载过程
类加载器还有双亲委派机制,这个就是三个加载器,
BootstrapClassLoader(启动类加载器)
c++
编写,加载java
核心库 java.*
,构造ExtClassLoader
和AppClassLoader
。由于引导类加载器涉及到虚拟机本地实现细节,开发者无法直接获取到启动类加载器的引用,所以不允许直接通过引用进行操作
ExtClassLoader (标准扩展类加载器)
java
编写,加载扩展库,如classpath
中的jre
,javax.*
或者java.ext.dir
指定位置中的类,开发者可以直接使用标准扩展类加载器。
AppClassLoader(系统类加载器)
java
编写,加载程序所在的目录,如user.dir
所在的位置的class
CustomClassLoader(用户自定义类加载器)
java
编写,用户自定义的类加载器,可加载指定路径的class
文件
打破双亲委派机制的方法就是重写loadclass方法不向上找,和findClass方法指定路径
双亲委派的好处就是java核心类不容易被篡改,然后还有一个沙箱安全机制就是java.lang.string,java 开头的有些核心包名的类禁止用自定义类加载器来加载