面试官:说一下构造函数,静态代码块,构造代码块等的加载顺序?
原创
©著作权归作者所有:来自51CTO博客作者wx6289ced28e34b的原创作品,请联系作者获取转载授权,否则将追究法律责任
介绍
构造函数,静态代码块,构造代码块的执行顺序
class Test {
// 静态代码块1
static {
System.out.println("我是静态代码块1");
}
// 构造代码块1:
{
System.out.println("我是构造代码块1");
}
// 构造函数1
public Test() {
System.out.println("我是无参构造函数");
}
// 构造函数2
public Test(int t) {
System.out.println("我是带参构造函数," + "参数是" + t);
}
// 静态代码块2
static {
System.out.println("我是静态代码块2");
}
// 构造代码块2:
{
System.out.println("我是构造代码块2");
}
}
来new一个Test类看看会输出什么?
public class FowShow {
public static void main(String[] args) {
// 创建第一个对象:
// 我是静态代码块
// 我是静态代码块2
// 我是构造代码块1
// 我是构造代码块2
// 我是无参构造函数
System.out.println("创建第一个对象:");
Test test1 = new Test();
// 创建第二个对象:
// 我是构造代码块1
// 我是构造代码块2
// 我是带参构造函数,参数是5
System.out.println("创建第二个对象:");
Test test2 = new Test(5);
}
}
从输出我们可以总结出如下结论
执行时间:静态代码块>构造代码块>构造函数
静态代码块只会在类被载入内存时加载一次,构造代码块和构造函数都是在对象创建的时候执行,有几个对象就会执行几次,所以一般将加载配置文件的过程写在静态代码块中
没有继承的加载顺序
先定义一个类作为成员变量,方便看打印输出
public class Parameter {
public Parameter(String str) {
System.out.println(str);
}
}
写一个测试类
public class Test1 {
public static Parameter parameter1 = new Parameter("静态成员变量");
public Parameter parameter2 = new Parameter("非静态成员变量");
public Test1() {
System.out.println("构造函数");
}
static {
System.out.println("静态代码块");
}
{
System.out.println("构造代码块a");
}
{
System.out.println("构造代码块b");
}
}
输出如下
public class ForShow {
public static void main(String[] args) {
/*
静态成员变量
静态代码块
非静态成员变量
构造代码块a
构造代码块b
构造函数
*/
new Test1();
/*
非静态成员变量
构造代码块a
构造代码块b
构造函数
*/
new Test1();
}
}
可以看到静态成员变量和静态代码块一样,只会在类被载入内存时加载一次
改变一下Test类中定义的顺序,看看有什么变化。
public class Test2 {
static {
System.out.println("静态代码块");
}
public Test2() {
System.out.println("构造函数");
}
{
System.out.println("构造代码块b");
}
{
System.out.println("构造代码块a");
}
public static Parameter parameter1 = new Parameter("静态成员变量");
public Parameter parameter2 = new Parameter("非静态成员变量");
}
}
输出如下
public class ForShow {
public static void main(String[] args) {
/*
静态代码块
静态成员变量
构造代码块b
构造代码块a
非静态成员变量
构造函数
*/
new Test2();
}
}
没有继承情况的加载顺序
- 静态代码块和静态成员变量,加载顺序由编写顺序决定
- 构造代码块和非静态成员变量,加载顺序由编写顺序决定
- 构造函数
有继承情况的加载顺序
public class Father {
public static Parameter parameter1 = new Parameter("父类静态成员变量");
public Parameter parameter2 = new Parameter("父类非静态成员变量");
public Father() {
System.out.println("父类的构造函数");
}
}
子类
public class Son extends Father{
public static Parameter parameter1 = new Parameter("子类静态成员变量");
public Parameter parameter2 = new Parameter("子类非静态成员变量");
public Son() {
System.out.println("子类的构造函数");
}
}
测试类
public class ForShow {
public static void main(String[] args) {
/*
父类静态成员变量
子类静态成员变量
父类非静态成员变量
父类的构造函数
子类非静态成员变量
子类的构造函数
*/
new Son();
}
}
有继承情况的加载顺序
- 父类的静态(静态代码块,静态成员变量),子类的静态(静态代码块,静态成员变量)
- 父类的非静态(构造代码块,非静态成员变量),父类的构造函数
- 子类的非静态(构造代码块,非静态成员变量),子类的构造函数
常见面试题
Java类定义如下,写出main函数执行后输出的结果
public class A {
static {
System.out.println("static A");
}
public A() {
System.out.println("class A");
}
}
public class B extends A{
public B() {
System.out.println("class B");
}
public static void main(String[] args) {
B instance = new B();
}
}
上面的知识点如果理解的很清楚的话,这个立马就能写出来。所以输出如下
2.问题如果main函数中的代码如下呢?就是B这个类new2次
public static void main(String[] args) {
B instance = new B();
B instance1 = new B();
}
万变不离其宗,输入如下
static A
class A
class B
class A
class B
因为静态成员变量和静态代码块只会在类被载入内存时加载一次
参考博客
构造函数执行时间