java编程思想读书笔记二(创建对象详解)

有兴趣的同学可以移步笔者的个人博客 更多博客

java对象

对象的创建

java的对象是在运行时创建的,创建对象的的触发条件有以下几种:

  1. 用new语句创建对象,这是最常用的创建对象方法。
  2. 运用反射手段,调用java.lang.reflect.Constructor类的newInstance()实例方法。
  3. 调用对象的clone()方法。
  4. 运用反序列化手段,调用java.io.ObjectInputStream对象的readObject()方法。

对象创建过程

java对象在创建时需要在方法区的运行时常量池去查找该类的符号引用,如果没有发现符号引用,说明该类还没有被JVM加载,所以要先进行JVM的加载。当JVM加载完时,会在java堆中分配内存。分配内存时会根据堆内存是否规整来分别进行两种方式的分配。
1.指针碰撞
把内存分为可用的和已用的,在中间放置一个指针,如果要分配对象的空间,则把内存的指针向可用的一端移动当前需要分配对象大小的距离。
2.空闲列表
当内存并不是很规整时,需要一个列表来维护那些列表是可用的,那些是不可用的。

当内存分配完成后需要对分配到内存空间的对象赋予零值(静态字段在类加载中就已经有值,所以不需要赋零值),接下来需要设置对象头的信息:如设置该对象的哈希码,属于哪个类,GC分代年龄等信息。从虚拟机的角度来看至此一个对象就创建完成,但是在java程序的角度看,对象的创建才刚刚开始,因为对象的值还没有设定,对象值得设定是由对象初始化化来完成的,初始化就是调用构造方法过程。

对象的初始化

当对象创建完成后,接下来就是进行对象的初始化了,也就是去执行构造方法。

父类的初始化

当子类对象创建之前首先会调用父类的构造函数,也就是会初始化父类,但是父类并没有被创建,也就是并没有在堆中给父类分配新的存储空间,而只是对父类的变量进行了赋值。从而达到子类可以使用父类的属性和方法的目的。

静态类的初始化

当一个类中有static修饰的方法或者是变量的话,那么当这个静态方法被第一次调用的时候,那么这个类就会初始化,就会调用此类的类构造器(在类加载过程中被JVM自动加入),类构造器只初始化一次,触发条件为实例构造器执行,或者是静态任何一个静态成员被引用,也就是说这个类只会初始化一次。

普通类的初始化

区别与父类和静态类的初始化,普通类的初始化是建立在对象创建之上的,也就是对象创建完成后会自动的去调用构造方法进行初始化。

对象创建及初始化实例

看完上面文字上的简单说明总感觉少些什么东西?就像一碗牛肉拉面没有卤鸡蛋一样。所以笔者要通过一个java代码来将上面的知识点串起来,让你有一个更清晰的认识。


public class Parent {
    int a = 1;
    static int b = 2;

    // 静态代码块
    static {
        System.out.println("执行Parent静态代码块:b =" + b);
        b++;
    }

    // 普通代码块
    {
        System.out.println("执行Parent普通代码块: a =" + a);
        System.out.println("执行Parent普通代码块: b =" + b);
        b++;
        a++;
    }

    // 无参构造函数
    Parent() {
        System.out.println("执行Parent无参构造函数: a =" + a);
        System.out.println("执行Parent无参构造函数: b =" + b);
    }

    // 有参构造函数
    Parent(int a) {
        System.out.println("执行Parent有参构造函数: a =" + a);
        System.out.println("执行Parent有参构造函数: b =" + b);
    }

    // 方法
    void fun() {
        System.out.println("执行Parent的fun方法");
    }

}

public class Child extends Parent {
    int c = 1;
    static int d = 2;
    // 静态代码块
    static {
        System.out.println("执行Child静态代码块:d =" + d);
        d++;
    }
    // 普通代码块
    {
        System.out.println("执行Child代码块: c =" + c);
        System.out.println("执行Child代码块: d =" + d);
        c++;
        d++;
    }

    // 构造函数
    Child() {
        System.out.println("执行Child构造函数: c =" + c);
        System.out.println("执行Child构造函数: d =" + d);
    }

    // 方法
    void fun() {
        System.out.println("执行Child的fun方法");
    }

}

public class Test {
    public static void main(String[] args) {
        Child demo = new Child();
        demo.fun();
        System.out.println("…………………………………………………………………………………………………………………………");
        Child child = new Child();
        child.fun();
    }
}

上面有三个很简单的类,一个Parent,一个Child,一个Test。当执行Test的main方法是会输出什么呢?

//输出结果
执行Parent静态代码块:b =2
执行Child静态代码块:d =2
执行Parent普通代码块: a =1
执行Parent普通代码块: b =3
执行Parent无参构造函数: a =2
执行Parent无参构造函数: b =4
执行Child代码块: c =1
执行Child代码块: d =3
执行Child构造函数: c =2
执行Child构造函数: d =4
执行Child的fun方法
…………………………………………………………………………………………………………………………
执行Parent普通代码块: a =1
执行Parent普通代码块: b =4
执行Parent无参构造函数: a =2
执行Parent无参构造函数: b =5
执行Child代码块: c =1
执行Child代码块: d =4
执行Child构造函数: c =2
执行Child构造函数: d =5
执行Child的fun方法

下面我们一起来看看这些输出是这么一步步产生的。



注:本文的重点不是虚拟机有关的详细执行,之后会专门写一个关于虚拟机加载的文章,所以有关细节问题都一笔带过了
欢迎大家来我的个人博客 http://anning.site

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章