Java-类加载器
1.类加载器
- 类加载器是当程序运行时要使用某个类,则类加载器就加该类的字节码加载到内存里执行。
- Java虚拟机可以安装多个类加载器,系统默认三个主要的类加载器,每个类加载器负责加载不同位置的类
- Bootstrap,ExtClassLoader,AppClassLoader。
- 有的类加载器也是Java类,所以必须有一个非Java类的加载器加载其他Java类的类加载器,这个就是Boostrap。
- Java虚拟机中的所有类加载器采用具有父子关系的树形结构进行组织。每个实例化类加载器对象时必须为其指定
- 一个父级的类加载器对象,或采用系统默认的类加载器作为父级。
2.ClassLoader
构造方法:
ClassLoader();//使用方法 getSystemClassLoader()
返回的 ClassLoader 创建一个新的类加载器,将该加载器作为父类加载器。
ClassLoader(ClassLoader parent);//指定父类加载器,父类可能最终调用Bootstrap作为最后的父级。
成员方法:
ClassLoader getParent();//返回委托的父类加载器。
static ClassLoader getSystemClassLoader();//返回系统的类加载器
Class<?> loadClass(String name);//使用指定的二进制名称来加载类。
Class<?> findClass(String name);//使用指定的二进制名称查找类.
Class<?> defineClass(byte [] b,int off ,int len);//读取byte数组转成类实例
3.三大类加载器的关系及加载类的范围
BootStrap(启动类加载器):
常用的Java类,如System,util下的集合类等等。
ExtClassLoader(扩展类加载器):
遵循双亲委派模型,即两种加载类的方式,一是loadClass方法,二是findClass方法。 它重写了findClass方法
当父类的loadClass方法找不到类,且它也找不到时,就会使用findClass方法加载类,还不行就会报异常错误。
我们可以通过右键该类export,将我们自定义的类导出到该类加载器的文件夹下,
AppClassLoader(应用程序类加载器):
没有遵循双亲委派模型,即一种加载类的方式,即loadClass方法,它重写了loadClass方法,
class NetworkClassLoader extends ClassLoader { String host; int port; public Class findClass(String name) { byte[] b = loadClassData(name); return defineClass(name, b, 0, b.length); } private byte[] loadClassData(String name) { // load the class data from the connection . . . } }
遵循双亲委派模型的意义:它保证了相同全限定名的类是不会被重复加载到JVM中,即没有重名的类加载上了。
不遵循双亲委派模型的意义:有可能有大量相同名的类,被不同的自定义类加载器加载到JVM中,即有同名不同功能的类都加载上来了
4.类加载器的(双亲)委托机制
类加载器加载类的时候,会委托父级加载器去找该类并加载该类,父类又委托给父类,直到祖宗加载该类,
祖宗没加载到该类的话,会让下一级找,直到最初委托的加载器(如果重写了findClass方法,则使用该方法加载类)。
还不行的话就会报异常ClassNotFoundException
- 首先当前线程的类加载器(
getContextClassLoader()
)去加载线程中的第一个类。 - 如果类A中引用了类B,Java虚拟机将使用加载类A的类装载器来加载类B。
- 还可以直接调用ClassLoader.loadClass()方法来指定某个类加载器去加载某个类。