Java类加载器
虚拟机自带的类加载器
- 启动类加载器(Bootstrap)C++
加载Java的核心类($JAVA_HOME/jre/lib/rt.jar),C++原生代码来实现 - 扩展类加载器(Extension)Java
加载jre扩展类($JAVA_HOME/jre/lib/ext/)或者由java.ext.dirs系统属性指定的目录中的JAR包的类。由Java语言实现,父类加载器为null(父类加载器为Bootstrap,null是因为通过java无法获取)。 - 应用程序类加载器(AppClassLoader)Java
也叫系统类加载器,加载当前应用的classpath的所有类
用户自定义加载器
- Java.lang.ClassLoader的子类,用户可以定制类的加载方式
JVM的类加载机制
- 双亲委派机制:
当一个类收到了类加载请求,它首先不会尝试自己去加载这个类,而是把这个请求委派给父类去完成,每一个层次类加载器都是如此,因此所有的加载请求都应该传送到启动类加载器中,只有当父类加载器反馈自己无法完成这个请求的时候(在它的加载路径下没有找到所需加载的Class),子类加载器才会尝试自己去加载。
采用双亲委派的一个好处是比如加载位于rt.jar包中的类java.lang.Object,不管是哪个加载器加载这个类,最终都是委托给顶层的启动类加载器进行加载,这样保证了使用不同的类加载器最终得到的都是同一个Object对象。
顺序:Bootstrap ==> Extension ==> AppClassLoader
简单的来说就是:先让父类加载,父类可以加载,成功返回,父类无法加载,自己再加载。
- 缓存机制:
缓存机制会将所有加载过的类存入缓存,当程序需要某个使用class时,类加载器会先查找缓存,当缓存中不存在的时候,类加载器才会去加载该class并存入缓存。
所以就算修改替换了class文件也要重启JVM服务才会生效哦!已经加载过的class存在缓存中不会再次被加载啦!
类加载过程
- 首先会查找缓存是否存在该class(缓存机制),如果存在直接返回对象,如果不存在,查找父类加载器(因为是通过加载器去查找对象,并且双亲委派机制,所以这里去查找父类加载器)。
- 如果父类加载器不存在,那么它的父类加载器即是根加载器(BootstrapClassLoader)或者本身就是根加载器,那么使用根加载器去查找目标类,如果成功则返回对象,失败报ClassNotFountException异常。
- 如果父类加载器存在,那么使用父类加载器去查找目标类,如果成功则返回对象,否则会通过当前类去查找目标类(同样双亲委派机制,父类无法加载则自己去加载),如果成功则返回对象,失败报ClassNotFountException异常。
网友评论