zoukankan      html  css  js  c++  java
  • 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

  • 相关阅读:
    static、final、this、super关键
    细节二:参数、引用类型、实例化
    枚举类型
    单例模式
    细节一:字符串、switch、默认值、数组
    类属性和类方法
    装饰器模式
    TreeSet
    可见参数和增强for以及自动拆装箱
    静态导入
  • 原文地址:https://www.cnblogs.com/anning1994/p/10028038.html
Copyright © 2011-2022 走看看