静态代码块:是用来初始化类的信息,在类被加载的时候就会被执行,且只执行一次。执行优先级高于非静态的初始化块。
非静态代码块:是用来初始化类的实例信息,在创建对象的时候就会被执行,且每创建一个对象都会被执行一次。执行的时候如果有静态初始化块,先执行静态初始化块再执行非静态初始化块,非静态初始化块会在构造函数执行时,在构造函数主体代码执行之前被运行。
构造方法:是用来创建对象的。
一、同类中静态代码块,非静态代码块,构造方法的执行顺序
执行顺序:静态代码块>非静态代码块>构造方法
即:程序在执行时,首先执行静态代码块,且只执行一次。接下来再创建对象时(new),首先执行非静态代码块,再执行构造方法。
示例1:
代码:
package com.csu.marden;
public class Demo {
static{
System.out.println("static");
}
{
System.out.println("no static");
}
Demo5(){
System.out.println("constructor");
}
public static void main(String[] args) {
}
}
执行结果:
分析:
上述Demo类包含静态代码块,非静态代码块和构造方法。但是该类中没有具体的执行,只完成了类的加载,即只有静态代码块在类加载的时候被执行,且只执行一次。由于没有对类的实例化,所有非静态代码块和构造方法均未执行。
示例2:
代码:
package com.csu.marden;
public class Demo5 {
static{
System.out.println("static");
}
{
System.out.println("no static");
}
Demo5(){
System.out.println("constructor");
}
public static void main(String[] args) {
Demo demo=new Demo();
}
}
执行结果:
分析:
上述Demo类包含静态代码块,非静态代码块和构造方法。上述代码首先完成了类加载,即静态代码块执行,且仅执行一次。然后创建了该类的实例,首先执行了非静态代码块,然后执行了构造方法。
示例3:
代码:
package com.csu.marden;
public class Demo5 {
static{
System.out.println("static");
}
{
System.out.println("no static");
}
Demo5(){
System.out.println("constructor");
}
public static void main(String[] args) {
Demo demo1=new Demo();
Demo demo2=new Demo();
}
}
执行结果:
分析:
上述Demo类包含静态代码块,非静态代码块和构造方法。上述代码首先完成了类加载,即静态代码块执行,且仅执行一次。然后创建了该类的两个实例,首先执行了非静态代码块,然后执行了构造方法。并且非静态代码块和构造方法,在每次创建新的对象时,都会被执行一次。
二、父类和子类中静态代码块,非静态代码块,构造方法的执行顺序
在Java继承中,父类子类构造方法、静态代码块、非静态代码块的执行顺序:
父类静态代码块 > 子类静态代码块 > 父类非静态代码块 > 父类构造方法 > 子类非静态代码块 > 子类构造方法
示例1:
代码:
//父类
public class Parent {
static String name = "hello";
{
System.out.println("parent-no-block");
}
static {
System.out.println("parent-static");
}
public Parent() {
System.out.println("parent-constructor");
}
}
//子类
public class Child extends Parent {
static String childName = "hello";
{
System.out.println("child-no-static");
}
static {
System.out.println("child-static");
}
public Child() {
System.out.println("child-constructor");
}
}
//测试类
public class Demo {
public static void main(String[] args) {
new Child();
}
}
执行结果:
分析:
Java继承体系中,若子类继承了父类,在创建子类的时候,首先会调用父类的构造方法和非静态代码块完成类实例的初始化,然后调用子类的构造方法和非静态代码块完成类实例的初始化(子类需要根据父类的构造方法生成)。
所以在执行时(创建子类对象时),首先加载父类,即执行父类的静态代码块,且只执行一次,然后加载子类,即执行子类的静态代码块,同样只执行一次。然后执行父类的非静态代码块和构造方法完成父类实例的初始化,最后执行子类的非静态代码块和构造方法完成子类实例的初始化。
示例2:
代码:
//父类
public class Parent {
static String name = "hello";
{
System.out.println("parent-no-block");
}
static {
System.out.println("parent-static");
}
public Parent() {
System.out.println("parent-constructor");
}
}
//子类
public class Child extends Parent {
static String childName = "hello";
{
System.out.println("child-no-static");
}
static {
System.out.println("child-static");
}
public Child() {
System.out.println("child-constructor");
}
}
//测试类
public class Demo {
public static void main(String[] args) {
new Child();
new Child();
}
}
执行结果:
分析:
在执行时(创建子类对象时),首先加载父类,即执行父类的静态代码块,且只执行一次,然后加载子类,即执行子类的静态代码块,同样只执行一次。然后执行父类的非静态代码块和构造方法完成父类实例的初始化,最后执行子类的非静态代码块和构造方法完成子类实例的初始化。由于本例中创建了两个子类对象,所以执行两次父类非静态代码块和构造函数,子类非静态代码块和构造函数。