写点什么

moon 不讲武德!!! 一个类加载机制给面试官说蒙了!!

用户头像
moon聊技术
关注
发布于: 2020 年 11 月 25 日
moon不讲武德!!!一个类加载机制给面试官说蒙了!!

正文约: 2900 字

预计阅读时间: 8 分钟

目录


  • 目录

  • 1 前言

  • 2 类加载机制 2.1 什么是类加载机制 2.2 案例 2.3 类加载的过程

  • 3 类加载器 3.1 什么是类加载器 3.2 双亲委派模型 3.3 破坏双亲委派模型

  • 4 结语


1 前言

    距离上次发表文章已经一周了,本来是打算早点肝出来的,但是由于不可抗力因素,年终了,需求急剧增加,再加上 moon 得给自己留出点学习时间,这篇文章也就拖到了现在,羞愧羞愧。

    今天我们来聊点基础却又不简单的东西,类加载机制,也是为 moon 的下一篇文章做个铺垫.

2 类加载机制

2.1 什么是类加载机制

    java 虚拟机把描述类的数据从 Class 文件加载到内存,并对数据进行校验、转换解析和初始化,最终形成可以被 jvm 可以直接使用的类型,这个过程就可以成为虚拟机的类加载机制。

2.2 案例

    我们先来看一个案例

public class World{  static{     System.ou.println("hello");  }  public static final String WORLD = "world";}public class Home extends World{  static{     System.ou.println("home");  }}public  class Test{  public static void main(String[] args){      System.out.println(Home.word);  }}
复制代码


     上面这个代码,究竟会输出什么?答案 moon 先告诉大家,最后的结果只会输出“word”,但是其中的原因你明白吗?我们接着往下看。

2.3 类加载的过程

     这是一张很经典的图,标明了一个类的生命周期,而很多人一眼看过去就以为明白了类的生命周期,但是这只是其中一种情况。

    真实情况是加载、验证、准备、初始化、卸载这五个阶段的顺序是确定的,是依次有序的。但是解析阶段有可能会在初始化之后才会进行,这是为了支持 java 动态绑定的特性。

    动态绑定: 在运行时根据具体对象的类型进行绑定。提供了一些机制,可在运行期间判断对象的类型,并分别调用适当的方法。也就是说,编译器此时依然不知道对象的类型,但方法调用机制能自己去调查,找到正确的方法主体。

    举个例子:

class a{    void test(){};  }  class b extends  a{    void test(){};  }  class c {    main(){      A a = new B();      a.test();    }  }
复制代码


    上述代码就可以很快的让你理解动态绑定了。

    A a = new B();这行代码在编译器的时候程序其实并不知道 new B()真正的引用是谁,在执行 a.test()时 ,也就是直到运行期间才确定,调用的是子类的 test(),其实这就是动态绑定。

    《java 虚拟机规范》规定,只有以下 6 种情况才会触发初始化:(以下参考《深入理解 java 虚拟机》)

  • 遇到 new、getstatic、putstatic 或 invokestatic 这 4 条字节码指令;

  • 使用 java.lang.reflect 包的方法对类进行反射调用的时候;(这里可能就会出现面试题,反射的缺点是什么

  • 当初始化一个类的时候,发现其父类还没有进行初始化的时候,需要先触发其父类的初始化;

  • 当虚拟机启动时,用户需要指定一个要执行的主类,虚拟机会先初始化这个类;

  • 当使用 JDK 1.7 的动态语言支持时,如果一个 java.lang.invoke.MethodHandle 实例最后的解析结果 REF_getStatic、REF_putStatic、REF_invokeStatic 的方法句柄,并且这个方法句柄所对应的类没有初始化。

  • 使用 java8 新加入的 default 默认方法,如果这个接口的实现类发生了初始化,那么该接口要在其之前被初始化。

    我们回到刚才的案例,正常来说当我们执行 Home.word 时,World 类就应该已经被加载了,但是关键点在于 word 这是个静态字段,而且 Home 这个类是继承了 World 类,而针对于静态变量,只有直接定义这个字段的类才会被初始化,所以说,如果这个静态变量没有被 final 修饰,那么正常情况下应该输出“hello”,“world",但是,由于是被 final 修饰的,就会在编译阶段直接存储在常量池中,最终调用的情况其实是 Test 类对常量池的引用,这下大家应该明白了。

3 类加载器

3.1 什么是类加载器

    实现"通过一个类的全限定名来获取描述该类的二进制流"的动作的代码就叫做类加载器。

    简单点来说,就是我知道你的名字后,我能知道你的全部,完成这个操作的就是"类加载器"。

3.2 双亲委派模型

    这是一张很经典的图,通常情况下,各个类加载器的协作关系就是这样的。

    双亲委派模型:简而言之,就是说一个类加载器收到了类加载的请求,不会自己先加载,而是把它交给自己的父类去加载,层层迭代

    用上图来说明就是如果应用程序类加载器收到了一个类加载的请求,会先给扩展类加载器,然后再给启动类加载器,如果启动类加载器无法完成这个类加载的请求,再返回给扩展类加载器,如果扩展类加载器也无法完成,就返回给应用类加载器。

    那么双亲委派模型的好处是什么?说这个问题前我要先和大家说一个概念,jvm 中类的唯一性是由类本身和加载这个类的类加载器决定的,简单的说,如果有个 a 类,如果被两个不同的类加载器加载,那么他们必不相等。你看到这里会不会想到所有类的父类都是 Object 是怎么实现的了吗?是因为无论哪一个类加载器加载 Object 类,都会交给最顶层的启动类加载器去加载,这样就保证了 Object 类在 jvm 中是唯一的。

3.3 破坏双亲委派模型

    当然,不是所有的类加载器都是遵循双亲委派模型的,和大家简单描述一个场景。

    我们在最初学习的时候肯定学习过 JDBC,其连接数据库的方式其实是通过一个 Driver 类去实现的,由于原生的 JDBC 中的类是放在 rt.jar 包的,是由启动类加载器进行类加载的,且需要动态去加载不同数据库类型的 Driver 类,而 mysql-connector-.jar 中的 Driver 类是用户自己写的代码,所以启动类加载器是不能进行加载的,需要由应用程序类加载器进行加载。此时,通过线程上下文类加载器获得应用程序类加载器,通过应用程序类加载器去加载这个 Driver 类,从而避开了双亲委派模型的局限性

4 结语

    其实这些东西都是死记硬背的东西,尤其是类加载的过程,其中有很多东西是没有什么值得关注的,只是为了应付面试,但是你需要明白的是为什么会这样设计,设计的好处是什么

    比如:

    为什么解析阶段有可能会在初始化阶段后才执行?

    双亲委派模型的好处是什么?为什么会这样设计?

    为什么会出现破坏双亲委派的模型?是解决了什么问题?

    大部分人学习一个新知识可能都是死记硬背,加上简单的理解,但是其实代码的世界很多地方都是互通的,要学会提取知识的精华,也就是思想,在自己的知识库中去训练一个模型,当你再学一个新知识的时候,很有可能你就会发现,这个知识,虽然我没有完全了解,但是它的设计思想我以前学过。真实的情况就是这样,尤其是你学到越多的框架,越多的技能,这种感知就会越来越深,一定要学会提炼,提炼,再提炼

    我是 moon,下期见,记得三连~


发布于: 2020 年 11 月 25 日阅读数: 57
用户头像

moon聊技术

关注

玩玩技术,聊聊人生,看看生活,搞搞理想 2019.03.19 加入

我是moon 文章首发于我的微信公众号:哪儿来的moon,欢迎大家关注 ! 关注后回复666 有一线大厂面试题赠送,助你成为offer收割机!

评论 (2 条评论)

发布
用户头像
我是moon
文章首发于我的微信公众号:moon聊技术,欢迎大家关注 !
有什么问题也可以在我的公众号联系我,和我直接交流~
公众号中也还有很多技术书文章,欢迎大家阅读、分享、转发~
关注后回复666 有一线大厂面试题赠送,助你成为offer收割机!
2020 年 12 月 03 日 15:52
回复
用户头像
我是moon
文章首发于我的微信公众号:哪儿来的moon,欢迎大家关注 !
有什么问题也可以从公众号和我直接交流~
关注后回复666 有一线大厂面试题赠送,助你成为offer收割机!
2020 年 11 月 25 日 11:30
回复
没有更多了
moon不讲武德!!!一个类加载机制给面试官说蒙了!!