服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - Java教程 - 分析Java中的类加载问题

分析Java中的类加载问题

2021-09-14 10:55华为云开发者社区 Java教程

很多时候提到类加载,大家总是没法马上回忆起顺序,这篇文章会用一个例子为你把类加载的诸多问题一次性澄清

一、Java类的加载顺序

引用1个网上的经典例子,并做稍许改动,以便大家更好地理解。

  1. public class Animal {
  2. private int i = test();
  3. private static int j = method();
  4. static {
  5. System.out.println("a");
  6. }
  7. Animal(){
  8. System.out.println("b");
  9. }
  10. {
  11. System.out.println("c");
  12. }
  13. public int test(){
  14. System.out.println("d");
  15. return 1;
  16. }
  17. public static int method(){
  18. System.out.println("e");
  19. return 1;
  20. }
  21. }
  22. public class Dog extends Animal{
  23. {
  24. System.out.println("h");
  25. }
  26. private int i = test();
  27. static {
  28. System.out.println("f");
  29. }
  30. private static int j = method();
  31.  
  32. Dog(){
  33. System.out.println("g");
  34. }
  35. public int test(){
  36. System.out.println("i");
  37. return 1;
  38. }
  39. public static int method(){
  40. System.out.println("j");
  41. return 1;
  42. }
  43. public static void main(String[] args) {
  44. Dog dog = new Dog();
  45. System.out.println();
  46. Dog dog1 = new Dog();
  47. }
  48. }

执行这段main程序,会输出什么?

答案是

eafjicbhig
icbhig

为了方便大家一个个细节去理解, 我换一种方式去提问。

Q: 什么时候会进行静态变量的赋值和静态代码块的执行?

A:

  • 第一次创建某个类或者某个类的子类的实例
  • 访问类的静态变量、调用类的静态方法
  • 使用反射方法forName
  • 调用主类的main方法(本例子的第一次静态初始化其实属于这个情况,调用了Dog的main方法)

注: 类初始化只会进行一次, 上面任何一种情况触发后,之后都不会再引起类初始化操作。

Q:初始化某个子类时,也会对父类做静态初始化吗?顺序呢?

A:如果父类之前没有被静态初始化过,那就会进行, 且顺序是先父类再子类。 后面的非静态成员初始化也是如此。
所以会先输出eafj。

Q: 为什么父类的method不会被子类的method重写?

A: 静态方法是类方法,不会被子类重写。毕竟类方法调用时,是必定带上类名的。

Q: 为什么第一个输出的是e而不是a?

A: 因为类变量的显示赋值代码和静态代码块代码按照从上到下的顺序执行。

Animal的静态初始化过程中,method的调用在static代码块之前,所以先输出e再输出a。

而Dog的静态初始化过程中,method的调用在static代码块之后,因此先输出f,再输出j

Q: 没有在子类的构造器中调用super()时,也会进行父类对象的实例化吗?

A: 会的。会自动调用父类的默认构造器。 super()主要是用于需要调用父类的特殊构造器的情况。
因此会先进行Animal的对象实例化,再进行Dog的对象实例化

Q: 构造方法、成员显示赋值、非静态代码块(即输出c和h的那2句)的顺序是什么?

A:

1.成员显示赋值、非静态代码块(按定义顺序)

2.构造方法

因此Animal的实例化过程输出icb(如果对输出i有疑问,见下面一题)
接着进行Dog的实例化,输出hig

Q: 为什么Animal实例化时, i=test()中输出的是i而不是d?

A:因为你真正创建的是Dog子类,Dog子类中的test()方法由于签名和父类test方法一致,因此test方法被重写了。
此时即使在父类中调用,也还是用使用子类Dog的方法。除非你new的是Animal。

Q: 同上题, 如果test方法都是private或者final属性, 那么上题的情况会有变化吗??

A:

因为private和final方法是不能被子类重写的。
所以Animal实例化时,i=test输出d。

总结一下顺序:

1.父类静态变量显式赋值、父类静态代码块(按定义顺序)

2.子类静态变量显式赋值、子类静态代码块(按定义顺序)

3.父类非静态变量显式赋值(父类实例成员变量)、父类非静态代码块(按定义顺序)

4.父类构造函数

5.子类非静态变量(子类实例成员变量)、子类非静态代码块(按定义顺序)

6.子类构造函数。

二、类加载过程

Q:类加载的3个必经阶段是:

A:

1.加载(类加载器读取二进制字节流,生成java类对象)

2.链接(验证,分配静态域初始零值)

3.初始化(前面的题目讲的其实就是初始化时的顺序)

更详细的如下:

分析Java中的类加载问题

三、被动引用中和类静态初始化的关系

Q:new某个类的数组时,会引发类初始化吗?

像下面输出什么

  1. public class Test {
  2. static class A{
  3. public static int a = 1;
  4. static{
  5. System.out.println("initA");
  6. }
  7. }
  8.  
  9. public static void main(String[] args) {
  10. A[] as = new A[5];
  11. }
  12. }

A:

new数组时,不会引发类初始化。
什么都不输出。

Q:引用类的final静态字段,会引发类初始化吗?

像下面输出什么?

  1. public class Test {
  2. static class A{
  3. public static final int a = 1;
  4. static{
  5. System.out.println("initA");
  6. }
  7. }
  8.  
  9. public static void main(String[] args) {
  10. System.out.println("A.a=" + A.a);
  11. }
  12. }

A: 不会引发。

不会输出initA。 去掉final就会引发了。
(注意这里必须是基本类型常量, 如果是引用类型产量,则会引发类初始化)

Q:子类引用了父类的静态成员,此时子类会做类初始化嘛?

如下会输出什么

  1. public class Test {
  2. static class A{
  3. public static int a = 1;
  4. static{
  5. System.out.println("initA");
  6. }
  7. }
  8.  
  9. static class B extends A{
  10. static {
  11. System.out.println("initB");
  12. }
  13. }
  14.  
  15. public static void main(String[] args) {
  16. System.out.println("B.a=" + B.a);
  17. }
  18. }

A:

子类不会初始化。
打印initA,却不会打印initB。

四、类加载器双亲委派

类加载时的双亲委派模型,不知道能怎么出题。。。反正就记得优先去父类加载器中看类是否能加载。

分析Java中的类加载问题

Bootsrap不是ClassLoader的子类,他是C++编写的。
而ExtClassLoader和AppClassLoader都是继承自ClassLoader的

Q:java中, 是否类和接口的包名和名字相同, 那么就一定是同一个类或者接口?

A:错误。

1个jvm中, 类和接口的唯一性由二进制名称以及它的定义类加载器共同决定。
因此2个不同的加载器加载出来相同的类或接口时, 实际上是不同的。

以上就是分析Java中的类加载问题的详细内容,更多关于Java 类加载的资料请关注服务器之家其它相关文章!

原文链接:https://www.cnblogs.com/huaweiyun/p/14921474.html

延伸 · 阅读

精彩推荐
  • Java教程Java BufferWriter写文件写不进去或缺失数据的解决

    Java BufferWriter写文件写不进去或缺失数据的解决

    这篇文章主要介绍了Java BufferWriter写文件写不进去或缺失数据的解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望...

    spcoder14552021-10-18
  • Java教程xml与Java对象的转换详解

    xml与Java对象的转换详解

    这篇文章主要介绍了xml与Java对象的转换详解的相关资料,需要的朋友可以参考下...

    Java教程网2942020-09-17
  • Java教程Java8中Stream使用的一个注意事项

    Java8中Stream使用的一个注意事项

    最近在工作中发现了对于集合操作转换的神器,java8新特性 stream,但在使用中遇到了一个非常重要的注意点,所以这篇文章主要给大家介绍了关于Java8中S...

    阿杜7472021-02-04
  • Java教程Java实现抢红包功能

    Java实现抢红包功能

    这篇文章主要为大家详细介绍了Java实现抢红包功能,采用多线程模拟多人同时抢红包,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙...

    littleschemer13532021-05-16
  • Java教程Java使用SAX解析xml的示例

    Java使用SAX解析xml的示例

    这篇文章主要介绍了Java使用SAX解析xml的示例,帮助大家更好的理解和学习使用Java,感兴趣的朋友可以了解下...

    大行者10067412021-08-30
  • Java教程升级IDEA后Lombok不能使用的解决方法

    升级IDEA后Lombok不能使用的解决方法

    最近看到提示IDEA提示升级,寻思已经有好久没有升过级了。升级完毕重启之后,突然发现好多错误,本文就来介绍一下如何解决,感兴趣的可以了解一下...

    程序猿DD9332021-10-08
  • Java教程20个非常实用的Java程序代码片段

    20个非常实用的Java程序代码片段

    这篇文章主要为大家分享了20个非常实用的Java程序片段,对java开发项目有所帮助,感兴趣的小伙伴们可以参考一下 ...

    lijiao5352020-04-06
  • Java教程小米推送Java代码

    小米推送Java代码

    今天小编就为大家分享一篇关于小米推送Java代码,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧...

    富贵稳中求8032021-07-12