写的java源文件必须经过编译和类加载两个阶段。
- 编译:把.java文件编译为.class文件的过程
- 类的加载过程:把.class 文件加载到JVM中
- 启动类加载器:加载%{Java_HOME}%\lib下面的 rt.jar 和 resource.jar
- 扩展类加载器:加载%{Java_HOME}%\lib\ext 下面的jar包和class文件
- 应用类加载器:加载当前应用里面的classpath下所有jar包和类文件
- 自定义类加载器:继承ClassLoader抽象类去实现自定义类的加载
双亲委派层级关系:
- 打破类加载器:类加载器加载不属于当前作用范围的类
打破类加载器的情况:
- 双亲委派是在JDK1.2版本发布的,而类加载器和抽象类ClassLoader是在JDK1.0就已经存在,用户可以重写ClassLoader里面的loadClass()方法去实现自定义类加载。JDK1.2里面为了向前兼容,需要去兼容loadClass()重写的一些实现;导致双亲委派机制被破坏。后序JDK版本不提倡实现loadClass()方法,而是实现findClass()方法,他能符合双亲委派机制。
- 存在顶层类加载器去调用底层类加载器的情况,比如说java.jdbc.Driver这样的一个接口,这个接口是由启动类加载器去加载,但是实现是由各大数据库厂商实现,应该由应用类加载器加载。JVM引入了线程上下文类加载器,它可以把原本是启动类加载器加载的类,转换为应用类加载器加载。
打破类加载器方法:
- 继承ClassLoad抽象类,重写loadClass()方法,方法里面可以自定义要加载的类使用的类加载器
- 使用线程的上下文加载器,可以通过java.lang.Thread类的setContextClassLoader()这个方法去设置当前使用的类加载器的一个类型