Java类的完整构造执行顺序

Python012

Java类的完整构造执行顺序,第1张

这里只说一个完整的结果 至于为什么是这样的顺序 可以参考我以前的文章 深入剖析java类的构造方式

如果父类有静态成员赋值或者静态初始化块 执行静态成员赋值和静态初始化块 如果类有静态成员赋值或者静态初始化块 执行静态成员赋值和静态初始化块 将类的成员赋予初值(原始类型的成员的值为规定值 例如int型为 float型为 f boolean型为false 对象类型的初始值为null) 如果构造方法中存在this()调用(可以是其它带参数的this()调用)则执行之 执行完毕后进入第 步继续执行 如果没有this调用则进行下一步 (这个有可能存在递归调用其它的构造方法) 执行显式的super()调用(可以是其它带参数的super()调用)或者隐式的super()调用(缺省构造方法) 此步骤又进入一个父类的构造过程并一直上推至Object对象的构造 执行类申明中的成员赋值和初始化块 执行构造方法中的其它语句

其中第 步是比较麻烦的 因为this调用实际上会调用类的另外一个构造方法 最终应该是执行类的某个构造方法 它可能会显示的调用super 但是无论是否调用super 最终都是执行super的 也就是父类的构造方法并一直这样递归到Object 所以在子类和父类的构造中 首先构造或者说执行的是父类的构造 但是它是由子类的构造方法调用的 先于构造方法的方法体里面的内容 这个是由编译器决定的 所以我感觉简单直观一些的顺序表述应该是

如果父类有静态成员赋值或者静态初始化块 执行静态成员赋值和静态初始化块 如果类有静态成员赋值或者静态初始化块 执行静态成员赋值和静态初始化块 将类的成员赋予初值(原始类型的成员的值为规定值 例如int型为 float型为 f boolean型为false 对象类型的初始值为null) 执行构造方法 并可能递归调用this() 最终先执行父类的构造方法并一直递归到Object的构造方法的执行 父类的构造方法执行完成后 执行类申明中的成员赋值和初始化块 执行构造方法中的其它语句

最终的简化顺序版本是

父类的静态成员赋值和静态块 子类的静态成员和静态块 父类的构造方法 父类的成员赋值和初始化块 父类的构造方法中的其它语句 子类的成员赋值和初始化块 子类的构造方法中的其它语句 lishixinzhi/Article/program/Java/hx/201311/27010

构造方法和实例方法的区别

一、主要的区别在于三个方面:修饰符、返回值、命名

1、和实例方法一样,构造器可以有任何访问的修饰符,public、private、protected或者没有修饰符   ,都可以对构造方法进行修饰。不同于实例方法的是构造方法不能有任何非访问性质的修饰符修饰,例如static、final、synchronized、abstract等都不能修饰构造方法。

解释:构造方法用于初始化一个实例对象,所以static修饰是没有任何意义的多个线程不会同时创建内存地址相同的同一个对象,所以synchronized修饰没有意义

构造方法不能被子类继承,所以final和abstract修饰没有意义。

2、返回类型是非常重要的,实例方法可以返回任何类型的值或者是无返回值(void),而构造方法是没有返回类型的,void也不行。

3、至于命名就是构造方法与类名相同,当然了实例方法也可以与类名相同,但是习惯上我们为实例方法命名的时候通常是小写的,另一方面也是与构造方法区分开。

而构造方法与类名相同,所以首字母一般大写。

下面看几个例子熟悉一下:

public class Sample {  

      

    private int x  

  

    public Sample() { // 不带参数的构造方法  

        this(1)  

    }  

      

    public Sample(int x) { //带参数的构造方法  

        this.x=x  

    }  

      

    public int Sample(int x) { //不是构造方法  

        return x++  

    }  

  

}

上面的例子即使不通过注释我们也很容易能区分开的,再看下面一个例子

public class Mystery {  

    private String s  

      

    public void Mystery() { //不是构造方法  

    s = "constructor"  

    }  

      

    void go() {  

    System.out.println(s)  

    }  

      

    public static void main(String[] args) {  

    Mystery m = new Mystery()  

    m.go()  

    }  

}

程序执行的结果为null,虽然说Mystery m = new Mystery()调用了Mystery 类的构造方法,但是public void Mystery()并不是构造方法,他只是一个普通的实例方法而已,那该类的构造方法哪去了呢?

二、说到这就得说一下java的默认构造方法

我们知道,java语言中规定每个类至少要有一个构造方法,为了保证这一点,当用户没有给java类定义明确的构造方法的时候,java为我们提供了一个默认的构造方法,这个构造方法没有参数,修饰符是public并且方法体为空。如果用户有定义构造方法,就不会有默认构造方法!!!

其实默认的构造方法还分为两种,一种就是刚刚说过的隐藏的构造方法,另一种就是显示定义的默认构造方法.

如果一个类中定义了一个或者多个构造方法,并且每一个构造方法都是带有参数形式的,那么这个类就没有默认的构造方法,看下面的例子。

public class Sample1{}  

  

public class Sample2{  

    public Sample2(int a){System.out.println("My Constructor")}  

}  

  

public class Sample3{  

    public Sample3(){System.out.println("My Default Constructor")}  

}

上面的三个类中Sample1有一个隐式的默认构造方法,下列语句Sample1 s1=new Sample()合法

Sample2没有默认的构造方法,下列语句Sample2 s2=new Sample2()不合法,执行会编译错误

Sample3有一个显示的默认构造方法,所以以下语句Sample3  s3=new Sample3()合法。

this代表你正在构造的对象。public class Test {public Test() {System.out.println(this.toString())}public static void main(String[] args) {Test obj = new Test()System.out.println(obj.toString())}}输出的两个字符串相同(后面的数字是对象地址),说明this和obj是同一个对象。这里this就是正在构建的CommunicationDemo的对象,作为参数传给Thread的构造函数。内幕是:java对象的建立其实是在构造函数开始执行之前,就是说构造函数执行的时候,被构造的对象其实已经存在了,所以构造函数中可以用this引用当前被构造的对象进行处理(比如this.name=“Jack”)。