关于classLoader类的说明
下图为classLoader类源码里的说明(JDK1.8)
class loader是一个负责加载classes的对象,ClassLoader类是一个抽象类,需要给出类的二进制名称,class loader尝试定位或者产生一个class的数据,一个典型的策略是把二进制名字转换成文件名然后到文件系统中找到该文件。
loadClass方法的具体实现:
进入方法通过加载操作的锁对象(通过getClassLoadingLock方法获取)来开启锁,调用findLoadedClass方法检查这个类是否被加载过,如果加载了,直接返回已加载的类。如果没加载,父类不为空委托给父类加载,使用父加载器调用loadClass方法,如果父类为Null,就检查是否是由启动类加载器加载的类,通过调用本地方法findBootstrapClass。如果任然没有返回类,类加载器装载虚拟机内置的加载器调用findClass(String)方法装载类。如果,按照以上的步骤成功的找到对应的类,并且该方法接收的resolve参数的值为true,那么就调用resolveClass(Class)方法来处理类。
分析该方法:该方法的访问控制符是protected,也就是说该方法同包内和派生类中可用,返回值类型Class。这里用到泛型。这里使用通配符?作为泛型实参表示对象可以 接受任何类型(类类型)。因为该方法不知道要加载的类到底是什么类,所以就用了通用的泛型。String name要查找的类的名字;boolean resolve,一个标志,true表示将调用resolveClass(c)处理该类。throws ClassNotFoundException 该方法会抛出找不到该类的异常,这是一个非运行时异常。
getClassLoadingLock方法的实现细节,我们看到这里用到变量parallelLockMap ,根据这个变量的值进行不同的操作,如果这个变量是Null,那么直接返回this,如果这个属性不为Null,那么就新建一个对象,然后在调用一个putIfAbsent方法来给刚刚创建好的对象赋值。
在ClassLoader类中有一个静态内部类ParallelLoaders,他会指定的类的并行能力,如果当前的加载器被定位为具有并行能力,那么他就给parallelLockMap定义,就是new一个ConcurrentHashMap<>(),那么这个时候,我们知道如果当前的加载器是具有并行能力的,那么parallelLockMap就不是Null,这个时候,判断parallelLockMap是不是Null,如果是null,说明该加载器没有注册并行能力,没有必要给他一个加锁的对象,getClassLoadingLock方法直接返回this,就是当前的加载器的一个实例。如果这个parallelLockMap不是null,那就说明该加载器是有并行能力的,那么就可能有并行情况,那就需要返回一个锁对象。然后就是创建一个新的Object对象,调用parallelLockMap的putIfAbsent方法,这个方法的作用是:根据传进来的className,检查该名字是否已经关联了一个value值,如果已经关联过value值,那么直接把他关联的值返回,如果没有关联过值的话,那就把我们传进来的Object对象作为value值,className作为Key值组成一个map返回。然后无论putIfAbsent方法的返回值是什么,都把它赋值给我们刚刚生成的那个Object对象。
resolveClass这个方法给Classloader用来链接一个类,如果这个类已经被链接过了,那么这个方法只做一个简单的返回。否则,这个类将被按照 Java™规范中的Execution描述进行链接。
native关键字说明:java平台有个用户和本地C代码进行互操作的API,称为Java Native Interface (Java本地接口)。详见:Java中Native关键字的作用。
总结: