博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Java类和对象初始化
阅读量:7125 次
发布时间:2019-06-28

本文共 2549 字,大约阅读时间需要 8 分钟。

类的生命周期:

Java类的初始化:

本阶段负责为类变量赋正确的初始值。(类变量即静态变量)

Java编译器把所有的类变量初始化语句和静态初始化器通通收集到<clinit>方法中,该方法只能被JVM调用,专门承担初始化工作。

初始化一个类必须保证其直接超类已被初始化。

并非所有类都拥有<clinit>()方法。以下类不会拥有<clinit>方法:

  1. 该类既没有声明任何类变量,也没有静态初始化语句。
  2. 该类声明了类变量,但没有使用类变量初始化语句或静态初始化语句初始化。
  3. 该类只包含静态final变量的类变量初始化语句,并且类变量初始化语句是常量表达式。

Java类初始化的时机:

规范定义类的初始化时机为“initialize on first active use”,即“在首次主动使用时初始化”。装载和链接在初始化之前就要完成。

首次主动使用的情形:

  1. 创建类的新实例--new,反射,克隆或反序列化;
  2. 调用类的静态方法;
  3. 操作类和接口的静态字段;(final字段除外)
  4. 调用Java的特定的反射方法;
  5. 初始化一个类的子类;
  6. 指定一个类作为Java虚拟机启动时的初始化类(含有main方法的启动类)。

除了以上6种情形,java中类的其他使用方式都是被动使用,不会导致类的初始化。

Java对象初始化:

编译器为每个类生成至少一个实例初始化方法,即<init>()方法。此方法与源程序里的每个构造方法对应。如果类没有声明构造方法,则生成一个默认构造方法,该方法仅调用父类的默认构造方法,同时生成与该默认构造方法对应的<init>()方法。

<init>()方法内容大概为:

  1. 调用另一个<init>()方法(本类的另外一个<init>()方法或父类的<init>()方法);
  2. 初始化实例变量;
  3. 与其对应的构造方法内的字节码

Java对象初始化的时机:

对象初始化又称为对象实例化。Java对象在其被创建时初始化。有两种方式创建Java对象:

一种是显示对象创建,通过new关键字来调用一个类的构造函数,通过构造函数创建一个对象。

一种是隐式对象创建:

  1. 加载一个包含String字面量的类或接口会引起一个新的String对象创建,除非包含相同字面量的String对象已经在JVM中存在了。
    String s1 = "zheng";

     

  2. 自动装箱机制可能会引起一个原子类型的包装类对象被创建。
    Integer iWrapper = 1;

     

  3. String连接符也可能会引起新的String或者StringBuilder对象被创建,同时还有可能引起原子类型的包装对象被创建。
    System.out.println("zheng"+1);

对象实例初始化的例子:

public class Base{    Base() {        preProcess();    }     void preProcess() {}}
public class Derived extends Base{    public String whenAmISet = "set when declared";     @Override void preProcess()    {        whenAmISet = "set in preProcess()";    }}
public class Main{    public static void main(String[] args)    {        Derived d = new Derived();        System.out.println( d.whenAmISet );    }}

下面是整个的运行流程:

  1. 进入Derived类构造函数
  2. Derived成员变量的内存被分配
  3. 调用Base类的构造函数
  4. Base类构造函数调用preProcess()方法
  5. Derived类的preProcess()方法设置whenAmISet
  6. Derived类的成员变量初始化被调用
  7. 执行Derived构造函数体

可以看到在执行完父类的构造函数后,第6步才对Derived类的成员变量初始化。

Java类初始化例子:

import java.util.Map;import java.util.HashMap;import java.util.Collections;public class Singleton {    public    static Singleton singleton = new Singleton();    public static Map m;    static{        m = new HashMap();    }    private Singleton(){        initM();    }    public static void initM(){        if(null == m){            System.out.println("m 为空");            m = new HashMap();        }        m.put("1", "郑");        m.put("2", "陈");    }    public static Singleton getInstance(){        return singleton;    }}
public class Main {    public static void main(String [] args){        Singleton singleton = Singleton.getInstance();    }}

运行结果输出m 为空

是因为在类初始化阶段先对singleton赋值调用Singleton类构造函数,然后Singleton类构造函数调用initM()方法。但是此时还没有运行static方法,所以m=null.

参考博客:

转载于:https://www.cnblogs.com/jxzheng/p/5191037.html

你可能感兴趣的文章
PHP设计模式
查看>>
聊聊flink TaskManager的managed memory
查看>>
oracle先排序再分页
查看>>
javascript 判断数据类型的几种方法
查看>>
前端进击的巨人(七):走进面向对象,原型与原型链,继承方式
查看>>
canvas 绘制贪吃蛇游戏
查看>>
时间复杂度与空间复杂度分析
查看>>
从Express到Nestjs,谈谈Nestjs的设计思想和使用方法
查看>>
Java的8中基本数据类型
查看>>
Android 原生app获取用户授权访问Autodesk云应用数据
查看>>
LeetCode 310. Minimum Height Trees
查看>>
《Java编程思想》读书笔记-对象导论
查看>>
计算机体系
查看>>
算法 - 时间复杂度
查看>>
如何截取视频片段 批量截取片段的方法
查看>>
OKR与Scrum如何强强联手
查看>>
dva中组件的懒加载
查看>>
IOS开发错误library not found for -lXXX
查看>>
Java动态追踪技术探究
查看>>
LeetCode刷题——29. Divide Two Integers(Part 1靠自己)
查看>>