分析Java中的类加载问题
作者:华为云开发者社区 发布时间:2023-09-03 19:37:04
一、Java类的加载顺序
引用1个网上的经典例子,并做稍许改动,以便大家更好地理解。
public class Animal {
private int i = test();
private static int j = method();
static {
System.out.println("a");
}
Animal(){
System.out.println("b");
}
{
System.out.println("c");
}
public int test(){
System.out.println("d");
return 1;
}
public static int method(){
System.out.println("e");
return 1;
}
}
public class Dog extends Animal{
{
System.out.println("h");
}
private int i = test();
static {
System.out.println("f");
}
private static int j = method();
Dog(){
System.out.println("g");
}
public int test(){
System.out.println("i");
return 1;
}
public static int method(){
System.out.println("j");
return 1;
}
public static void main(String[] args) {
Dog dog = new Dog();
System.out.println();
Dog dog1 = new Dog();
}
}
执行这段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.初始化(前面的题目讲的其实就是初始化时的顺序)
更详细的如下:
三、被动引用中和类静态初始化的关系
Q:new某个类的数组时,会引发类初始化吗?
像下面输出什么
public class Test {
static class A{
public static int a = 1;
static{
System.out.println("initA");
}
}
public static void main(String[] args) {
A[] as = new A[5];
}
}
A:
new数组时,不会引发类初始化。
什么都不输出。
Q:引用类的final静态字段,会引发类初始化吗?
像下面输出什么?
public class Test {
static class A{
public static final int a = 1;
static{
System.out.println("initA");
}
}
public static void main(String[] args) {
System.out.println("A.a=" + A.a);
}
}
A: 不会引发。
不会输出initA。 去掉final就会引发了。
(注意这里必须是基本类型常量, 如果是引用类型产量,则会引发类初始化)
Q:子类引用了父类的静态成员,此时子类会做类初始化嘛?
如下会输出什么
public class Test {
static class A{
public static int a = 1;
static{
System.out.println("initA");
}
}
static class B extends A{
static {
System.out.println("initB");
}
}
public static void main(String[] args) {
System.out.println("B.a=" + B.a);
}
}
A:
子类不会初始化。
打印initA,却不会打印initB。
四、类加载器双亲委派
类加载时的双亲委派模型,不知道能怎么出题。。。反正就记得优先去父类加载器中看类是否能加载。
Bootsrap不是ClassLoader的子类,他是C++编写的。
而ExtClassLoader和AppClassLoader都是继承自ClassLoader的
Q:java中, 是否类和接口的包名和名字相同, 那么就一定是同一个类或者接口?
A:错误。
1个jvm中, 类和接口的唯一性由二进制名称以及它的定义类加载器共同决定。
因此2个不同的加载器加载出来相同的类或接口时, 实际上是不同的。
来源:https://www.cnblogs.com/huaweiyun/p/14921474.html


猜你喜欢
- 我就废话不多说了,大家还是直接看代码吧~ public static void main(String[] args) { &n
- 问题分析疑惑满满小枫听到这个面试题的时候,心想这是什么水面试官,怎么问这么简单的题目,心想一个for循环加上equal判断再删除不就完事了吗
- 一.线程不安全多线程的执行环境中,程序的执行结果和预期的结果不符合,这就称为发生了线程不安全现象二.那些情况导致了线程不安全?大致分为以下5
- 重写java object类的equals方法覆盖equals方法请遵守约定什么情况下要覆盖equals方法容易违反的对称性不易察觉的传递性
- gradle文件执行流程做过Android开发的同学都知道 ,Android项目中存在三个gradle文件,那你是否知道他们的执行流程呢?请
- 前言Java虽然五脏俱全但总有软肋,譬如获取CPU等硬件信息,当然我们可以通过JNI调用C/C++来获取,但对于对C/C++和Windows
- 1.@RequestMapping注解1.1@RequestMapping注解的功能从注解名称上我们可以看到,@RequestMapping
- 前言身在孤岛有很多无奈,比如说程序员属于比较偏门的职业。尤其是早些年,在行业里跳过几次槽后,可能你就已经认识整个圈子的人了。然后,再跳槽很可
- @Value注解读取yml中的map配置网上查了好多资料,都是.properties文件中读取,而且又是几个人抄来抄去,找了半天功夫不负有心
- 在界面设计中,一个容器要放置许多组件,为了美观,为组件安排在容器中的位置,这就是布局设计。java.awt中定义了多种布局类,每种布局类对应
- 上标是指比同一行中其他文字稍高的文字,而下标是指比同一行中其他文字稍低的文字。在生活中,我们常见的平方米、立方米等符号以及化学中的各种元素符
- Spring核心Spring核心是 IOC 和 AOP 。所谓IoC,对于spring框架来说,就是由spring来负责控制对象的生命周期和
- 一、什么是IO流输入流和输出流。输入流:数据从数据源(文件)到程序(内存)的路径输出流:数据从程序(内存)到数据源(文件)的路径二、常用的文
- spring-MVC实现简单的登录功能,供大家参考,具体内容如下今天我学习了spring-MVC实现简单的登录功能,本篇博客就讲解如何使用s
- 概述Spring Boot简化了Spring应用的开发过程,遵循约定优先配置的原则提供了各类开箱即用(out-of-the-box)的框架配
- 一、首先我们要获取Logcat中的日志如何获取呢?首先我们要先定义一个String[]数组,里面的代码是//第一个是Logcat ,也就是我
- 今天使用Android Studio建了一个新项目,遇到了不少问题,网上也找了不少解决方案都无效(可能Studio版本、项目版本等都不一样,
- JavaWeb项目部署到服务器详细步骤本地准备在eclipse中将项目打成war文件:鼠标右键要部署到服务器上的项目导出项目数据库文件MyS
- C语言数据结构之二叉树的非递归后序遍历算法前言:前序、中序、后序的非递归遍历中,要数后序最为麻烦,如果只在栈中保留指向结点的指针,那是不够的
- 一直在使用Mybatis这个ORM框架,都是使用mybatis里的一些常用功能。今天在项目开发中有个业务是需要限制各个用户对某些表里的字段查