做网站排名多少钱,上海设计网站开发,域名注册的网站,长沙seo排名优化公司转载自 关于Java类加载双亲委派机制的思考#xff08;附面试题#xff09; 预定义类加载器和双亲委派机制
JVM预定义的三种类型类加载器#xff1a;
启动#xff08;Bootstrap#xff09;类加载器#xff1a;是用本地代码实现的类装入器#xff0c;它负责将 Java_R…转载自 关于Java类加载双亲委派机制的思考附面试题 预定义类加载器和双亲委派机制
JVM预定义的三种类型类加载器
启动Bootstrap类加载器是用本地代码实现的类装入器它负责将 Java_Runtime_Home/lib下面的类库加载到内存中比如rt.jar。由于引导类加载器涉及到虚拟机本地实现细节开发者无法直接获取到启动类加载器的引用所以不允许直接通过引用进行操作。标准扩展Extension类加载器是由 Sun 的 ExtClassLoadersun.misc.Launcher$ExtClassLoader实现的。它负责将 Java_Runtime_Home /lib/ext或者由系统变量 java.ext.dir指定位置中的类库加载到内存中。开发者可以直接使用标准扩展类加载器。系统System类加载器是由 Sun 的 AppClassLoadersun.misc.Launcher$AppClassLoader实现的。它负责将系统类路径CLASSPATH中指定的类库加载到内存中。开发者可以直接使用系统类加载器。
除了以上列举的三种类加载器还有一种比较特殊的类型 — 线程上下文类加载器。
双亲委派机制描述 某个特定的类加载器在接到加载类的请求时首先将加载任务委托给父类加载器依次递归如果父类加载器可以完成类加载任务就成功返回只有父类加载器无法完成此加载任务时才自己去加载。
几点思考
Java虚拟机的第一个类加载器是Bootstrap这个加载器很特殊它不是Java类因此它不需要被别人加载它嵌套在Java虚拟机内核里面也就是JVM启动的时候Bootstrap就已经启动它是用C写的二进制代码不是字节码它可以去加载别的类。
这也是我们在测试时为什么发现System.class.getClassLoader()结果为null的原因这并不表示System这个类没有类加载器而是它的加载器比较特殊是BootstrapClassLoader由于它不是Java类因此获得它的引用肯定返回null。
委托机制具体含义 当Java虚拟机要加载一个类时到底派出哪个类加载器去加载呢
首先当前线程的类加载器去加载线程中的第一个类假设为类A。 注当前线程的类加载器可以通过Thread类的getContextClassLoader()获得也可以通过setContextClassLoader()自己设置类加载器。如果类A中引用了类BJava虚拟机将使用加载类A的类加载器去加载类B。还可以直接调用ClassLoader.loadClass()方法来指定某个类加载器去加载某个类。
委托机制的意义 — 防止内存中出现多份同样的字节码 比如两个类A和类B都要加载System类
如果不用委托而是自己加载自己的那么类A就会加载一份System字节码然后类B又会加载一份System字节码这样内存中就出现了两份System字节码。如果使用委托机制会递归的向父类查找也就是首选用Bootstrap尝试加载如果找不到再向下。这里的System就能在Bootstrap中找到然后加载如果此时类B也要加载System也从Bootstrap开始此时Bootstrap发现已经加载过了System那么直接返回内存中的System即可而不需要重新加载这样内存中就只有一份System的字节码了。
一道面试题
能不能自己写个类叫java.lang.System
答案通常不可以但可以采取另类方法达到这个需求。解释为了不让我们写System类类加载采用委托机制这样可以保证爸爸们优先爸爸们能找到的类儿子就没有机会加载。而System类是Bootstrap加载器加载的就算自己重写也总是使用Java系统提供的System自己写的System类根本没有机会得到加载。
但是我们可以自己定义一个类加载器来达到这个目的为了避免双亲委托机制这个类加载器也必须是特殊的。由于系统自带的三个类加载器都加载特定目录下的类如果我们自己的类放在一个特殊的目录那么系统的加载器就无法加载也就是最终还是由我们自己的加载器加载。