Java类加载机制
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
Java类加载机制
java类加载机制
类是java编程语⾔的基本单元。
java的源代码经过编译后⽣成java的字节码⽂件(class⽂件),字节码⽂件是以⼆进制的形式存储。
在运⾏时,这些类的字节码⽂件会加载进⼊JVM的内存的元空间中,并且以Class<T>的形式对类进⾏描述。
本⽂将详细讲解java的类加载机制。
类加载流程
加载:通过classloader将字节码⽂件以⼆进制字节流的形式读⼊到内存中,将字节流转换为⽅法区运⾏时的数据结构,在内存中⽣成⼀个Class<T>对象对类进⾏描述。
链接:验证阶段检查字节码⽂件是否符合JVM规范,准备阶段为类中的静态字段分配内存并赋予初始值,解析阶段将虚拟机中常量池中的符号引⽤转化为直接引⽤。
符号引⽤存在于编译⽣成的字节码中,⽤来描述当前类对其他类的引⽤。
直接引⽤是可以是直接指向⽬标的指针、相对偏移量或是⼀个能间接定位到⽬标的句柄。
解析阶段也可以在运⾏过程中发⽣,这个跟动态语⾔调⽤相关。
初始化:初始化是类加载的最后⼀步,前⾯的类加载的过程中,除了在加载阶段⽤户应⽤程序可以通过⾃定义类加载器参与外,其余动作完全由虚拟机主导和控制。
到了初始化阶段,才真正开始执⾏类中定义的java程序代码。
它主要是负责:初始化阶段是执⾏类构造器(静态代码块)< clinit >()⽅法的过程,< clinit >()是编译器⾃动收集类中所有的类变量的赋值动作、静态代码块产⽣的。
ClassLoader
ClassLoader顾名思义是类的加载器,类的加载要通过ClassLoader进⾏,ClassLoader的职责是将字节码⽂件从磁盘或者⽹络中加载进JVM内存。
同时你也可以在java代码中操作ClassLoader定义⼀些⾃定义的⾏为。
ClassLoader是⼀个抽象基类,你可以继承它重写⾃⼰⾃定义的加载流程。
⼀般我们的java类会通过⼏个常见的类加载器加载,它们分
为BootstrapClassLoader,ExtensionClassLoader,ApplicaitonClassLoader。
BootstrapClassLoader主要加载的是JVM⾃⾝需要的类,这个类加载使⽤C++语⾔实现的,是虚拟机⾃⾝的⼀部分,它负责将<JAVA_HOME>/lib路径下的核⼼类库或-Xbootclasspath参数指定的路径下的jar包加载到内存中,注意必由于虚拟机是按照⽂件名识别加载jar包的,如rt.jar,如果⽂件名不被虚拟机识别,即使把jar包丢到lib⽬录下也是没有作⽤的(出于安全考虑,Bootstrap启动类加载器只加载包名为java、javax、sun等开头的类)。
ExtensionClassLoader是指Sun公司(已被Oracle收购)实uncher$ExtClassLoader类,由Java语⾔实现的,是Launcher的静态内部类,它负责加载<JAVA_HOME>/lib/ext⽬录下或者由系统变量-Djava.ext.dir指定位路径中的类库,开发者可以直接使⽤标准扩展类加载器。
ApplicationClassLoader也称应⽤程序加载器是指 Sun公司实现的uncher$AppClassLoader。
它负责加载系统类路径java -classpath或-D java.class.path指定路径下的类库,也就是我们经常⽤到的classpath路径,开发者可以直接使⽤系统类加载器,⼀般情况下该类加载是程序中默认的类加载器,通过ClassLoader#getSystemClassLoader()⽅法可以获取到该类加载器。
双亲委派机制
双亲委派机制是指,当⼀个ClassLoader尝试加载⼀个类时,它并不会⾃⼰加载,⽽是将加载任务向上委托给⽗加载器加载。
每个ClassLoader中都有⼀个parent属性,⽤来保存⽗加载器的引⽤。
注意:⽗加载器并不是⽗类加载器,它们之间没有类之间的继承关系。
双亲委派机制的加载流程为:在类加载器缓存中查询,此类是否已经加载,若已加载,则直接由此加载器加载,若没有则向上委托给⽗加载器加载。
若最上层⽗加载器也未加载,则向下委托给⼦加载器加载。
protected Class<?> loadClass(String name, boolean resolve)
throws ClassNotFoundException
{
synchronized (getClassLoadingLock(name)) {
// ⾸先检查类是否已经加载过
Class<?> c = findLoadedClass(name);
if (c == null) {
// 没有加载过,则委托⽗加载器加载
long t0 = System.nanoTime();
try {
if (parent != null) {
// 若有⽗加载器,则委托⽗加载加载
c = parent.loadClass(name, false);
} else {
// 若没有⽗加载器(最上层⽗加载器也未加载此类),则委托BootStrap加载器加载
c = findBootstrapClassOrNull(name);
}
} catch (ClassNotFoundException e) {
// ClassNotFoundException thrown if class not found
// from the non-null parent class loader
}
if (c == null) {
// If still not found, then invoke findClass in order
// to find the class.
// 这种情况代表Bootstrap加载器也未加载此类,则委托给本加载器加载。
long t1 = System.nanoTime();
c = findClass(name);
// this is the defining class loader; record the stats
PerfCounter.getParentDelegationTime().addTime(t1 - t0);
PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
PerfCounter.getFindClasses().increment();
}
}
if (resolve) {
resolveClass(c);
}
return c;
}
测试双亲委派机制
新建⼀个测试⽤的类
package misc;
public class Model
{
static
{
System.out.println("类被加载了");
}
public static void sayHello()
{
System.out.println("hello");
}
}
⾃⼰定义⼀个类加载器
public class MyClassLoader extends ClassLoader
{
private final String classPath;
public MyClassLoader(String classPath)
{
this.classPath = classPath;
}
@Override
protected Class<?> findClass(String name) throws ClassNotFoundException
{
// 如果使⽤MyClassLoader加载,那么这句话将会输出⾄控制台
System.out.println("使⽤MyClassLoader加载");
var classFilePath = classPath + "/" + name.replace(".","/").concat(".class");
var bao = new ByteArrayOutputStream();
var readByte = 1;
try
{
var fis = new FileInputStream(classFilePath);
while ((readByte = fis.read()) != -1)
{
bao.write(readByte);
}
var bytesArray = bao.toByteArray();
return defineClass(name, bytesArray, 0, bytesArray.length);
} catch (IOException ex)
{
ex.printStackTrace();
throw new ClassNotFoundException(ex.getMessage());
}
}
}
测试代码
public static void loadClassViaMyClassLoader() throws Exception
{
var classLoader = new MyClassLoader("/Users/huobingnan/code/java/misc/out/production/misc");
var modelClass = classLoader.loadClass("misc.Model");
System.out.println("Model的类加载器是:" + modelClass.getClassLoader());
var sayHelloMethod = modelClass.getDeclaredMethod("sayHello");
sayHelloMethod.invoke(null);
}
测试输出
Model的类加载器是:jdk.internal.loader.ClassLoaders$AppClassLoader@7c53a9eb
类被加载了
hello
通过输出可以发现,Model类的加载并未使⽤我们⾃定义的MyClassLoader,⽽是使⽤了JDK中的应⽤程序类加载器,这就是双亲委派机制的体现,你也可以对上述代码进⾏DEBUG运⾏,从中便可得知类加载的途径是
MyClassLoader -> AppClassLoader -> PlatformClassLoader -> BootstrapClassLoader -> PlatformClassLoader -> AppClassLoader。
注意:不同的JDK可能加载器的名称会有所不同,笔者这⾥使⽤的是zulu-jdk-arm64。
打破双亲委派机制
通过上⽂的测试⽤例可以得知,尽管我们⾃定义了ClassLoader,但是由于双亲委派机制的存在,字节码⽂件没有使⽤我们⾃定义的ClassLoader加载。
那么如何强制字节码⽂件使特定ClassLoader加载呢?我们可以通过重写CLassLoader.loadClass(String name, boolean resovle)⽅法进⾏。
例如下⾯的代码:
public class MyClassLoader extends ClassLoader
{
private final String classPath;
public MyClassLoader(String classPath)
{
this.classPath = classPath;
}
@Override
protected Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException
{
if (name.startsWith("misc"))
{
// 如果包名以misc开头,我们使⽤MyClassLoader加载
return findClass(name);
}
return super.loadClass(name, resolve);
}
@Override
protected Class<?> findClass(String name) throws ClassNotFoundException
{
System.out.println("使⽤MyClassLoader加载");
var classFilePath = classPath + "/" + name.replace(".", "/").concat(".class");
var bao = new ByteArrayOutputStream();
var readByte = 1;
try
{
var fis = new FileInputStream(classFilePath);
while ((readByte = fis.read()) != -1)
{
bao.write(readByte);
}
var bytesArray = bao.toByteArray();
return defineClass(name, bytesArray, 0, bytesArray.length);
} catch (IOException ex)
{
ex.printStackTrace();
throw new ClassNotFoundException(ex.getMessage());
}
}
}
更改之后,再次使⽤上⽂中的测试代码进⾏测试:
使⽤MyClassLoader加载
Model的类加载器是:misc.MyClassLoader@d041cf
类被加载了
hello
这⾥可以看到,通过重写loadClass⽅法,我们可以⾃定义类加载⾏为,打破双亲委派机制。
打破双亲委派机制带来的问题
虽然我们⾃定义了类加载,并且打破了双亲委派机制,使得我们可以⾃定义类加载器加载类的⾏为。
但是打破双亲委派机制后会带⼀个问题:
public static void testClassLoaderCastBehavior() throws Exception
{
var classLoader = new MyClassLoader("/Users/huobingnan/code/java/misc/lib");
var modelClass = classLoader.loadClass("misc.Model");
var object = modelClass.getConstructor().newInstance();
var model = (Model)object;
}
使⽤MyClassLoader加载
类被加载了
Exception in thread "main" ng.ClassCastException: class misc.Model cannot be cast to class misc.Model (misc.Model is in unnamed module of loader misc.MyClassLoader @d041cf; misc.Model is in unnamed module of loader 'app') at misc.ClassLoaderTest.testClassLoaderCastBehavior(ClassLoaderTest.java:19)
at misc.ClassLoaderTest.main(ClassLoaderTest.java:85)
使⽤⾃定义类加载器,在⽂件系统中加载了⼀个类,这个类与我们项⽬类路径中的Model类定义完全⼀致,但是他们之间并不能进⾏强制类型转换。
这也就是说,虽然我们可以加
载这个类,但是在使⽤的时候只能通过反射的⽅式进⾏。
我们知道通过反射对⼀个类进⾏操作会带来隐患,⽽且对于⽤户来说,这样的调⽤操作并不直观。
同时,这种⾏为也限制了我们在项⽬中保留接⼝定义的情况下,⽆法通过类加载器的加载实现类并强制转换使⽤。
如果想要通过接⼝的形式进⾏上述操作需要借助java的SPI机制。
参考⽂献
1. 张善⾹. 解析Java虚拟机开发:权衡优化,⾼效和安全的最优⽅案[M]. 北京: 清华⼤学出版社: 2013.
2. java类加载机制(全套)
3. Java SPI机制。