前言

之前详细介绍了Java类的整个加载过程(类加载机制详解)。虽然,篇幅较长,但是也不要被内容吓到了,其实每个阶段都可以用一句话来概括。

1)加载:查找并加载类的二进制字节流数据。

2)验证:保证被加载的类的正确性。

3)准备:为类的静态变量分配内存,并设置默认初始值。

4)解析:把类中的符号引用转换为直接引用。

5)初始化:为类的静态变量赋予正确的初始值。

当然,要想掌握类加载机制,还是需要去深入研究的。(好吧,说了一句正确的废话)因为其中,有很多知识点也是面试中常问的。比如,我之前去面试的时候,面试官就问到了一个和类初始化相关的问题。就是给一段代码,有父子类关系,父子类中包含静态代码块、构造代码块、普通代码块、构造函数等,然后让判断代码最终的执行顺序。(可自行思考一下,具体内容细节暂时不做扩展)

类加载器

终于来到了本文的主题 —— 类加载器和双亲委派机制。

在《深入理解Java虚拟机》中,对于类加载器的定义是这样的:

虚拟机设计团队把类加载阶段中的“通过一个类的权限定名来获取描述此类的二进制字节流”这个动作放到Java虚拟机外部去实现,以便让应用程序自己决定如何去获取所需要的类。实现这个动作的代码模块称为“类加载器”。

简单来说,类加载器的作用就是去加载class类的二进制字节流的。

类加载器有以下三种:

1)启动类加载器(Bootstrap ClassLoader),或者叫根加载器。这个类加载器主要是去加载你在本机配置的环境变量 Java_Home/jre/lib 目录下的核心API,如rt.jar

2)扩展类加载器(Extension ClassLoader)。这个加载器负责加载 Java_Home/jre/lib/ext 目录下的所有jar包。

3)应用程序类加载器(Application ClassLoader)。这个加载器加载的是你的项目工程的ClassPath目录下的类库。如果用户没有自定义自己的类加载器,这个就是程序默认的类加载器。

另外,如果有需要的话,用户也可以自定义自己的类加载器(去继承ClassLoader类)。

我们也可以通过代码把类加载器打印出来:

public class TestClassLoader {

public static void main(String[] args) {

Object obj = new Object();

System.out.println(obj.getClass().getClassLoader());

TestClassLoader t = new TestClassLoader();

System.out.println(t.getClass().getClassLoader());

System.out.println(t.getClass().getClassLoader().getParent());

System.out.println(t.getClass().getClassLoader().getParent().getParent());

}

}

打印结果:

null

sun.misc.Launcher$AppClassLoader@58644d46

sun.misc.Launcher$ExtClassLoader@6d6f6e28

null

注意,上面第一行和第四行的null此处可不是空的意思,它代表的是启动类加载器。因为启动类加载器是用C++代码来实现的,严格来说不属于Java类,所以Java代码访问不到,故返回null。第二行是应用程序类加载器,第三行是扩展类加载器。

双亲委派机制

在介绍双亲委派机制之前,先观察一下以下代码能否正确运行:

//自己定义的一个 java.lang包

package java.lang;

public class String {

public static void main(String[] args) {

String s = new String();

System.out.println(s);

}

}

以上代码,编译没有任何问题,但是运行时,却报错:

为什么提示在java.lang.String类中找不到main方法呢,我这明明不是定义了吗?其实,问题的关键就在于类加载遵循双亲委派机制。

类加载器有以下这样的层次关系:

当一个类在加载的时候,都会先委派它的父加载器去加载,这样一层层的向上委派,直到最顶层的启动类加载器。如果顶层无法加载(即找不到对应的类),就会一层层的向下查找,直到找到为止。 这就是类的双亲委派机制。

这样做有什么好处呢?这就相当于维护了一个有优先级的层级关系,即总是从最顶层的父加载器开始加载。这就如同,你工作中遇到了问题需要向上反馈,比如先反馈给小组长,然后小组长反馈给上级经理,最后经理反馈给boss。然后boss感觉这问题太简单了不需要他亲自出手,让经理自己解决吧,然后经理又向下交给小组长。小组长一看,这问题不算难,人也比较热心,于是就帮你把问题解决了。(可能例子不是太恰当哈,意思理解即可)

到此,我们就明白了为什么上边的代码会报错。因为双亲委派机制的存在,去加载我们自己定义的“java.lang.String”类的时候,会最终委派到顶层的启动类加载器,然后找到了rt.jar包下的“java.lang.String”。找到之后,就直接加载rt.jar包的String类(也就是我们经常使用的那个字符串类),不再去向下查找,也就加载不了我们自定义的String类了。由于,rt.jar包下的String类中确实没有main方法,所以才会有以上的报错信息。

我们可以试想一下,如果没有双亲委派机制的存在,那我这段代码是不是就可以执行成功了。如果这样的话,岂不是说明我可以随意覆盖rt.jar包中的类(如String,Integer类等)。这样的话将会使程序陷入混乱,Java核心包中的类的安全也无法保证。

java 类加载 双亲委派_Java类加载器和双亲委派机制相关推荐

  1. java 类加载 双亲委派_java类加载器和双亲委派模型

    一. 类加载器 ClassLoader即常说的类加载器,其功能是用于从Class文件加载所需的类,主要场景用于热部署.代码热替换等场景. 系统提供3种的类加载器:Bootstrap ClassLoad ...

  2. java classloader详解_Java类加载器(ClassLoader)详解

    本文主要讲述Java ClassLoader的工作原理,这为后面将Android App代码热替换或者插件化升级做铺垫 一. 类加载器 ClassLoader即常说的类加载器,其功能是用于从Class ...

  3. java log4j 热部署_JAVA类加载器分析--热部署的缺陷(有代码示例,及分析)

    首先纠正一个误区: 热部署不是我们在Eclipse里面修改了代码不用重启就可以持续调试. 详情可以参考上一篇博文: 热部署和"Hot Code Replace"的区别    htt ...

  4. java class加载_Java 类加载

    从一个诡异的问题说起 测试案例一: packageecut.classloader;public classSun {protected static int a = 100;protected st ...

  5. java记录类加载的个数_java类加载过程 - audience_1的个人空间 - OSCHINA - 中文开源技术交流社区...

    java类加载过程: 在我们代码中,一般就一个new的方法,就能够创建一个java的对象类,但在java虚拟机中 1:首先,我们需要判断该类是否已经在方法区中的常量池中有一个类的符号的引用存在,如果该 ...

  6. java多个构造方法_Java构建器(多个构造器参数)

    今天看netty权威指南,第一次听说构建器,百度了几个博客,但是并没有通俗易懂一点儿的,综合别人的博客,总结如下: 1. 构建器是什么? 当创建对象需要传入多个参数的时候我们通常会根据参数的数量写不同 ...

  7. java 反射与泛型_Java基础系列 - 泛型和反射机制

    package com.test5; import java.lang.reflect.Field; import java.lang.reflect.Method; /** * Java泛型和反射机 ...

  8. java 打印预览_java打印和打印预览机制实例代码

    代码如下: import java.awt.basicstroke; import java.awt.borderlayout; import java.awt.color; import java. ...

  9. java回调与异步_Java同步异步玩转回调机制

    北京这周终于凉快了不少,舒了一口气,但是中暑带来的后遗症还没减轻,晕晕乎乎地上了一周的班也不见好转,基本就是这状态: 程序员 好了,不扯了.上篇文章讲了回调的基本原理:调用方通过方法将自己身的实例传给 ...

最新文章

  1. 中国开源大爆发进行时,你没掉队吧?
  2. 【Python刷题】_2
  3. 算法与数据结构(快速排序)
  4. SQL基础【十九、触发器】(不建议使用触发器的原因)
  5. verilog对YCrCb转换灰度设计及仿真
  6. 超标量、超级流水线、超长指令字、向量机 SIMD
  7. RDLC系列之五 初试XAML
  8. 优化Meta讨好搜索引擎 更好的提升网站排名
  9. java 接口 设计模式吗_JAVA接口设计模式-工厂模式
  10. python中使用函数的目的是什么_Python代码中os.mknod()函数起什么作用呢?
  11. easyUI之ComboBox(下拉列表框)
  12. demo 网络运维_【运维面试】面试官:你们公司的docker主要用来做啥?
  13. 35-BigDecimal详解
  14. Class的三种构造方法
  15. 软件测试实例——总结
  16. RubyOnRails with Ajax
  17. 计算机网络是几级学科,教育部更新学科目录 “网络空间安全”增设为一级学科...
  18. 批量处理word文档向下箭头
  19. 如何在页面直接打开微信公众号
  20. A20.从零开始前后端react+flask - 查找数据

热门文章

  1. Ubuntu16版本安装截图软件Flameshot
  2. python对Excel的操作 xlrd、xlwt包详解
  3. 进入opencv内部函数调试
  4. 实现两个点集的欧式距离和cos距离和索引值寻找(含有两种解法,for循环和矩阵操作)
  5. Android之Intent深入
  6. Python进阶之“属性(property)”详解
  7. Lambda 表达式详解~Lambda与集合
  8. java 操作db2_java操作DB2,增删改查
  9. java中final是修饰符么_Java final修饰符详解
  10. 北大暑期课作业 - 对cnblog 和其他技术博客的分析,比较和展望