结论
-
构造器:和类同名而且没有返回值,用来初始化类属性(自然是对象实例化一次就执行一次)
- 代码块的作用:用来初始化类、对象的信息
-
构造代码块:对象实例化一次,就执行一次;
- 内部可以输出语句
- 随着对象的创建而执行
- 每创建一个对象,就执行一次非静态代码块
- 作用:可以在创建对象时,对对象的属性等进行初始化
- 如果一个类中定义了多个非静态代码块,则按照声明的先后顺序执行
- 非静态代码块内可以调用静态的属性、静态的方法,或非静态的属性、非静态的方法
- 静态代码块:对象实例化就执行,无论new多少个对象,仅仅执行一次;
- 静态代码块是跟着类的加载而加载的,而构造代码块和构造方法是随着对象的加载而加载的
- 内部可以输出语句
- 随着类的加载而执行,而且只执行一次
- 作用:初始化类的信息
- 如果一个类中定义了多个静态代码块,则按照声明的先后顺序执行
- 静态代码块的执行要优先于非静态代码块的执行
- 静态代码块内只能调用静态的属性、静态的方法,不能调用非静态的结构
执行顺序
静态代码块------》构造代码块-----》构造函数
如果涉及到继承
父类的静态代码块------》子类的静态代码块------》父类的构造代码块------》父类的构造函数-------》子类的构造代码块--------》子类的构造函数
下面贴代码解释:
执行结构:
静态代码块 null
构造代码块Test@19db3e20 null
构造器Test@19db3e20 123
从结果可以看出:
静态代码块先执行,name是null,构造代码块,name是null,构造器Test里面是123
可以看出在构造函数之前执行可构造代码块.
如果调用new Test两次。
执行结果如下
静态代码块 null
构造代码块Test@759f6a57 null
构造器Test@759f6a57 123
构造代码块Test@34feb215 123
构造器Test@34feb215 123
可以看出静态代码块只会执行一次,
构造代码块和构造器能对此执行,并且构造代码块在构造器之前执行
关于继承中的子类和父类相关代码块和函数的执行顺序
代码
class Father
{
static int num;
static String name;
static{
System.out.println("父类的静态代码块"+num+" "+name);
}
{
System.out.println("父类的构造代码块+"+num+" "+name);
}
Father(){
System.out.println("父类的无参构造方法"+num+" "+name);
}
}
class Child extends Father
{
static{
System.out.println("子类的静态代码块"+num+" "+name);
}
{
System.out.println("子类的构造代码块+"+num+" "+name);
}
Child(){
System.out.println("子类的无参构造方法"+num+" "+name);
}
Child(int num,String name){
this.num=num;
this.name=name;
System.out.println("子类的有参构造方法"+this.num+" "+this.name);
}
}
public class Test
{ public static void main(String [] args){
Child c=new Child();
Child c1=new Child(1,"ls");
}
}
结果
**父类的静态代码块0 null
子类的静态代码块0 null
父类的构造代码块+0 null
父类的无参构造方法0 null
子类的构造代码块+0 null
子类的无参构造方法0 null
父类的构造代码块+0 null
父类的无参构造方法0 null
子类的构造代码块+0 null
子类的有参构造方法1 ls**
A:静态随着类的加载而加载。
B:静态代码块 -- 构造代码块 -- 构造方法的执行流程
静态代码块 -- 构造代码块 -- 构造方法
C:只要有子父关系,肯定先初始化父亲的数据,然后初始化子类的数据。
除了上述代码块,还有局部代码块,和同步代码块