网站首页 > java教程 正文
-----------------第二天------------------------
本文先论述父子类变量、代码块、构造函数执行顺序的结论, 然后通过举例论证,接着再扩展,彻底搞懂静态代码块、动态代码块、构造函数、父子类、类加载机制等知识体系。温故而知新,建议点赞收藏~
1 先说结论
面试官:好的,你说一下java中父类子类的变量、代码块、构造函数执行顺序是怎样的?
3妹:好的,
Java程序中类中个元素的初始化顺序 初始化的原则是:
- 先初始化静态部分,再初始化动态部分,
- 先初始化父类部分,后初始化子类部分,
- 先初始化变量,再初始化代码块和构造器。
具体的,由于类中初始化的部分有静态成员变量、静态代码块、普通成员变量、动态代码块、构造函数。所以跟父类子类组合起来有10种情况,总体顺序是:
- 1.父类的静态成员变量(如果是第一次加载类)
- 2.父类的静态代码块(如果是第一次加载类)
- 3.子类的静态成员变量(如果是第一次加载类)
- 4.子类的静态代码块(如果是第一次加载类)
- 5.父类的普通成员变量
- 6.父类的动态代码块
- 7.父类的构造器方法
- 8.子类的普通成员变量
- 9.子类的动态代码块
- 10.子类的构造器方法
2 举例
面试官:能写个栗子,验证一下吗
3妹:好的,假设有一个Parent和Child类
public class Parent {
static Instance staticInstance = new Instance("1---Parent类的静态成员变量staticInstance");
static {
System.out.println("2---Parent类的静态代码块执行了");
}
Instance instance = new Instance("5---Parent类的普通成员变量instance");
{
System.out.println("6---Parent类的动态代码块执行了");
}
Base() {
System.out.println("7---Parent类的构造器方法执行了");
}
}
public class Child extends Parent {
static Instance staticInstance = new Instance("3---Child类的静态成员变量staticInstance");
static {
System.out.println("4---Child类的静态代码块执行了");
}
Instance instance = new Instance("8---Child类的普通成员变量instance");
{
System.out.println("9----Child类的动态代码块执行了");
}
Child() {
System.out.println("10---Child类的构造器方法执行了");
}
//执行测试
public static void main(String[] args) {
Child child = new Child();
}
}
测试结果如下:
1---Parent类的静态成员变量staticInstance进行了初始化
2---Parent类的静态代码块执行了
3---Child类的静态成员变量staticInstance进行了初始化
4---Child类的静态代码块执行了
5---Parent类的普通成员变量instance进行了初始化
6---Parent类的动态代码块执行了
7---Parent类的构造器方法执行了
8---Child类的普通成员变量instance进行了初始化
9----Child类的动态代码块执行了
10---Child类的构造器方法执行了
说明确实是按照上面的执行顺序执行的。
3 静态变量、静态代码块
面试官:好的,我们先来说一个静态变量和静态代码块,他们的加载时机是怎样的
3妹:
静态变量、静态代码块是被static修饰的,是属于当前类的信息,类加载过程是先将编译后的class文件加载到内存中,一个类只会被加载到内存中一次。而static修饰的代码块属于类的信息的,所以,静态代码块中的代码有且只有一次被执行。执行的时机:类被加载的时候。
4 动态代码块
面试官:那动态代码块呢?
3妹:
动态代码块即不是static修饰的代码块,是用来初始化类实例信息的。当我们new关键字创建一个对象的时候,就会被执行,而且每使用一个new关键字创建出一个新对象的时候就会被执行一次的,在构造函数主题代码执行之前被运行的。
5 父类与子类的构造函数
面试官:再来说一下构造函数,new 一个子类的时候,父类的构造函数是何时被加载的呢?
3妹:
父类与子类的加载时机:父类在子类前面
需要注意的是:子类的构造方法,不管是无参构造还是有参构造,默认都会先去寻找父类的无参构造方法。如果父类中,没有无参构造,那么子类必须使用supper这个关键字来调用父类带参数的构造方法,否则在编译期都不能通过。
如下图:
6 类的加载及初始化
面试官:再来说一下类的加载及初始化是怎样的?
3妹:好的:
Java类从被加载到虚拟机内存中开始,到卸载出内存为止,它的整个生命周期包括:加载(Loading)、验证(Verification)、准备(Preparation)、解析(Resolution)、初始化(Initialization)、使用(Using) 和 卸载(Unloading)七个阶段。其中准备、验证、解析3个部分统称为连接(Linking)。
类加载的时机:
什么情况下虚拟机需要开始加载一个类呢?虚拟机规范中并没有对此进行强制约束,这点可以交给虚拟机的具体实现来自由把握。
类初始化则是类加载的最后一步,
类初始化的时机:
虚拟机规范指明 有且只有 五种情况必须立即对类进行初始化(而这一过程发生在加载、验证、准备之后):
- 1、遇到new、getstatic、putstatic或invokestatic这四条字节码指令(注意,newarray指令触发的只是数组类型本身的初始化,而不会导致其相关类型的初始化,比如,new String[]只会直接触发String[]类的初始化,也就是触发对类[Ljava.lang.String的初始化,而直接不会触发String类的初始化)时,如果类没有进行过初始化,则需要先对其进行初始化。生成这四条指令的最常见的Java代码场景是:
1)使用new关键字实例化对象的时候;
2) 读取或设置一个类的静态字段(被final修饰,已在编译器把结果放入常量池的静态字段除外)的时候;
3)调用一个类的静态方法的时候。 - 2、使用java.lang.reflect包的方法对类进行反射调用的时候,如果类没有进行过初始化,则需要先触发其初始化。
- 3、当初始化一个类的时候,如果发现其父类还没有进行过初始化,则需要先触发其父类的初始化。
- 4、当虚拟机启动时,用户需要指定一个要执行的主类(包含main()方法的那个类),虚拟机会先初始化这个主类。
- 5、当使用jdk1.7动态语言支持时,如果一个java.lang.invoke.MethodHandle实例最后的解析结果REF_getstatic,REF_putstatic,REF_invokeStatic的方法句柄,并且这个方法句柄所对应的类没有进行初始化,则需要先出触发其初始化。
7 再次总结一下
面试官:好的, 今天的面试就到这里,很高兴你能来参加面试~
3妹:谢谢面试官,再见~
3妹:今天的面试面试的还挺细的,我要总结一下:
Java程序中类中个元素的初始化顺序 初始化的原则是:
- 先初始化静态部分,再初始化动态部分,
- 先初始化父类部分,后初始化子类部分,
- 先初始化变量,再初始化代码块和构造器。
具体的,由于类中初始化的部分有静态成员变量、静态代码块、普通成员变量、动态代码块、构造函数。所以跟父类子类组合起来有10种情况,总体顺序是:
- 1.父类的静态成员变量(如果是第一次加载类)
- 2.父类的静态代码块(如果是第一次加载类)
- 3.子类的静态成员变量(如果是第一次加载类)
- 4.子类的静态代码块(如果是第一次加载类)
- 5.父类的普通成员变量
- 6.父类的动态代码块
- 7.父类的构造器方法
- 8.子类的普通成员变量
- 9.子类的动态代码块
- 10.子类的构造器方法
还有静态代码块、动态代码块、构造函数、父子类、类加载、类初始化,这些知识点虽然普通,但是也要牢记哦~
猜你喜欢
- 2024-10-17 Java|剖析类内的五类成员:属性、方法、构造器、代码块、内部类
- 2024-10-17 C++|构造、析构、成员(静态、非静态)函数、友元、全局函数
- 2024-10-17 JavaScript构造函数和原型:构造函数和原型 原创
- 2024-10-17 灵魂拷问:创建 Java 字符串,用""还是构造函数
- 2024-10-17 dart系列之:dart类中的构造函数(dart命名构造函数)
- 2024-10-17 Java面试题#构造函数和Setter哪种DI方式更好
- 2024-10-17 「php」construct() 函数介绍与使用方法详解
- 2024-10-17 为什么不允许使用 Java 静态构造函数?
- 2024-10-17 关于构造函数的Java面试问题(关于构造函数的java面试问题有哪些)
- 2024-10-17 编程语言Java如何创建对象,看完秒懂
你 发表评论:
欢迎- 07-15采用Oracle OSB总线进行服务注册和接入
- 07-15javaEE 新闻管理系统 oracle11+tomcat6
- 07-15从Oracle演进看数据库技术的发展(oracle数据库发展史)
- 07-15如何升级oracle数据库安全补丁(oraclepsu补丁升级)
- 07-15【权威发布】关于Oracle WebLogic Server未授权远程代码执行高危漏洞的预警通报
- 07-15【mykit-data】 数据库同步工具(数据库表同步工具)
- 07-15[Java速成] 数据库基础,Connector/J、JDBC、JPA的关系(day 7)
- 07-15Google前工程主管“入住”Oracle(google浏览器找不到以前的书签)
- 最近发表
-
- 采用Oracle OSB总线进行服务注册和接入
- javaEE 新闻管理系统 oracle11+tomcat6
- 从Oracle演进看数据库技术的发展(oracle数据库发展史)
- 如何升级oracle数据库安全补丁(oraclepsu补丁升级)
- 【权威发布】关于Oracle WebLogic Server未授权远程代码执行高危漏洞的预警通报
- 【mykit-data】 数据库同步工具(数据库表同步工具)
- [Java速成] 数据库基础,Connector/J、JDBC、JPA的关系(day 7)
- Google前工程主管“入住”Oracle(google浏览器找不到以前的书签)
- Oracle数据库云服务系列新增前所未有的企业级功能
- 直播预告丨如何实现Oracle存储过程到java的一键转化
- 标签列表
-
- java反编译工具 (77)
- java反射 (57)
- java接口 (61)
- java随机数 (63)
- java7下载 (59)
- java数据结构 (61)
- java 三目运算符 (65)
- java对象转map (63)
- Java继承 (69)
- java字符串替换 (60)
- 快速排序java (59)
- java并发编程 (58)
- java api文档 (60)
- centos安装java (57)
- java调用webservice接口 (61)
- java深拷贝 (61)
- 工厂模式java (59)
- java代理模式 (59)
- java.lang (57)
- java连接mysql数据库 (67)
- java重载 (68)
- java 循环语句 (66)
- java反序列化 (58)
- java时间函数 (60)
- java是值传递还是引用传递 (62)
本文暂时没有评论,来添加一个吧(●'◡'●)