Java对象创建过程补遗

一、static修饰的东东是属于这个类的,是所有的该类的实例共享的,因此它们的初始化先于实例对象的初始化。

二、Java中没有静态构造方法,但是有静态代码块。当类中同时存在静态代码块和静态成员变量声明语句时,它们谁先执行?

答:按照类中定义的从上到下顺序,谁在前面谁先执行。

三、当类中同时存在非静态代码块和非静态成员变量声明语句时,它们谁先执行?

答:按照类中定义的从上到下顺序,谁在前面谁先执行。

四、含有赋值的非静态成员变量声明语句,如 int v=4,其实这个语句的执行要分成两步进行。首先创建对象时初始化变量v,此时v=0;然后在构造器中再给v赋值,也就是v=4。

五、非静态初始化块中的赋值语句、实例变量值初始化赋值语句,都比构造器中的赋值语句先执行。

原理如下:不管是非静态初始化块中的赋值语句,还是实例变量值初始化赋值语句,它们在编译器处理后,都会被提取到构造器中,而且在构造器中原有的语句之前。

public class JavaTest
{
    //成员变量
    int count=20;

    //非静态初始化代码块
    {
        count = 12;
        System.out.print("占个位置而已");
    }

    //构造器
    public JavaTest()
    {
        System.out.println(count);
    }

    public JavaTest(String name)
    {
        System.out.println(count);
    }
}

编译后等价于:

public class JavaTest
{
    //成员变量
    int count;

    //非静态初始化代码块
    {
        System.out.print("占个位置而已");
    }

    //构造器
    public JavaTest()
    {
        count=20;//原来的成员变量赋值
        count=12;//原来的初始化代码块赋值
        System.out.println(count);
    }

    public JavaTest(String name)
    {
        count=20;//原来的成员变量赋值
        count=12;//原来的初始化代码块赋值
        System.out.println(count);
    }
}

六、“Java对象是由构造器创建的”这句话是错误的。构造器只是负责对Java对象的实例变量执行初始化而已,就像前面的三、四、五点所提及的。在执行构造器代码之前,该实例对象所占的内存已经被分配下来了,这些内存都默认是空值——对于基本类型的变量,默认的空值就是0、0.0或者false;对于引用类型的变量,默认的控制就是Null。看看下面的代码,分析一下结果是什么:

class base
{
    private int i = 2;

    public Base()
    {
        这里是1处
        this.display();
    }

    public void display()
    {
        System.out.println(i);
    }
}

class Derived extends base
{
    private int i = 22;

    public Derived()
    {
        i = 222;
    }

    public void display()
    {
        System.out.println(i);
    }
}

public class Test
{
    public static void main(String[] args)
    {
        new Derived();
    }
}

问题一:Derived对象的实例中有几个”i“实例变量? 说只有一个的同学,请自己去面壁

问题二:代码中的红色的1处,所指的“this”是谁?是base的实例还是Derived的实例?  答案是 Derived

问题三:代码中的红色1处,“this.display(); ”调用的是哪个类里面的display方法?       答案是 Derived

问题四:执行Test后输出是2?22?还是222?   答案是 0

七、编译器在处理成员变量和方式时是有区别的。当子类和父类存在同名的成员变量时,实际是各自拥有一个名字相同的成员变量;当子类中有个方法和父类签名相同的时候,实际是子类对父类中该方法的重写,这个时候通过子类的实例调用这个名称的方法将始终是子类的方法,哪怕你把子类强制转换成父类。这么说有点迷糊,直接往下看:

class base
{
    private int i = 2;

    public Base()
    {
        this.display();
    }

    public void display()
    {
        System.out.println("我是父类"+i);
    }
}

class Derived extends base
{
    private int i = 22;

    public Derived()
    {
        i = 222;
    }

    public void display()
    {
        System.out.println("我是子类"+i);
    }
}

情况一:

Base b=new Base()

b.count :肯定是2,因为当前对象是父类的实例

b.display():肯定是调用父类的方法,因为当前是父类的实例

情况二:

Derived d=new Derived();

b.count: 肯定是22,因为当前对象是子类的实例,子类自己有个成员变量叫“i”干嘛拿父类的

b.display():子类的方法

情况三:

Base bd=new Derived();

b.count:肯定是2,因为当前子类是被向上转换了,不得不拿父类的“i”

b.display():还是子类的方法,向上转换不能太过分了,子类已经重写的方法不能再反悔了。要想调用父类的同名方法,还得用这种方式:super.display()

情况四:

Derived d=new Derived();

Base d2b=d;

b.count:肯定是2,因为当前子类是被强制类型转换了,不得不拿父类的“i”

b.display():还是子类的方法,强制类型转换也不能太过分了,子类已经重写的方法不能再反悔了。要想调用父类的同名方法,还得用这种方式:super.display()

八、从内存分配的角度来看,当程序创建一个子类对象时,JVM不仅会为该类中定义的实例变量分配内存,同时也会为其父类中定义的所有实例变量分配内存,即使子类定义了与父类中同名的实例变量。如果子类中定义了与父类同名的成员变量,那么子类中的变量会隐藏父类中定义的变量,但不是完全覆盖,只是不能直接用而已;当把子类转换成父类的时候或者使用”super“关键字,就可以使用父类中的同名成员变量了。

时间: 2024-10-13 07:11:29

Java对象创建过程补遗的相关文章

Java 对象创建过程

在某些项目中,会使用到静态块,构造器中初始化其他对象...深刻理解了java类初始化过程,这些将不在是问题. 那java对象的步骤是什么呢? 假设现在有People类,未显示继承任何其他类,初始化过程是这样的. 1 java 解释器必须查找类路径,以定位People.class文件.在首次创建对象时(构造器是静态方法),或people类的静态方法/静态域首次被访问时. 2 载入People.class,静态初始化的所有动作都会执行,且静态初始化只在Class对象首次加载的时候进行一次. 3 当使

Java对象创建过程

假设有个名为Dog的类 1.当首次创建型为Dog的对象时(构造器可以看成静态方法),或者Dog类的静态方法/静态域首次被访问时,Java解释器必须查找类路径,以定位Dog.class文件. 2.然后载入Dog.class(这将创建一个Class对象),有关静态初始化的动作都会执行.因此,静态初始化只在Class对象首次加载的时候进行一次. 3.当你用new Dog()创建对象的时候,首先将在堆上为Dog对象分配足够的存储空间. 4.这块存储空间会被清零,这就自动地将Dog中的所有基本类型数据设置

Java中对象创建过程

本文介绍的对象创建过程仅限于普通Java对象,不包括数组和Class对象. 1.类加载检查 虚拟机遇到一条new指令时,首先去检查该指令的参数能否在常量池中定位到一个类的符号引用,并且检查这个符号引用代表的类是否已被加载.解析和初始化过. 2.为新生对象分配内存 对象所需内存的大小在类加载完成后便可以完全确定,为对象分配空间的任务等同于把一块确定大小的内存从Java堆中划分出来.划分的方法主要有两种: a.指针碰撞 如果java堆中内存是绝对规整的,所有用过的内存都放在一边,未使用的内存放在另一

深入理解java虚拟机(二)HotSpot Java对象创建,内存布局以及访问方式

内存中对象的创建.对象的结构以及访问方式. 一.对象的创建 在语言层面上,对象的创建只不过是一个new关键字而已,那么在虚拟机中又是一个怎样的过程呢? (一)判断类是否加载.虚拟机遇到一条new指令的时候,首先会检查这个指令的参数是否能在常量池中定位到一个类的符号引用,并且检查这个符号代表的类是否被加载.解析并初始化.如果没有完成这个过程,则必须执行相应类的加载. (二)在堆上为对象分配空间.对象需要的空间大小在类加载完成后便能确定.之后便是在堆上为该对象分配固定大小的空间.分配的方式也有两种:

深度理解java虚拟机读书笔记(二)HotSpot Java对象创建,内存布局以及访问方式

内存中对象的创建.对象的结构以及访问方式. 一.对象的创建 在语言层面上,对象的创建只不过是一个new关键字而已,那么在虚拟机中又是一个怎样的过程呢? (一)判断类是否加载.虚拟机遇到一条new指令的时候,首先会检查这个指令的参数是否能在常量池中定位到一个类的符号引用,并且检查这个符号代表的类是否被加载.解析并初始化.如果没有完成这个过程,则必须执行相应类的加载. (二)在堆上为对象分配空间.对象需要的空间大小在类加载完成后便能确定.之后便是在堆上为该对象分配固定大小的空间.分配的方式也有两种:

Java对象创建阶段的代码调用顺序

在创建阶段系统通过下面的几个步骤来完成对象的创建过程 为对象分配存储空间 开始构造对象 从超类到子类对static成员进行初始化 超类成员变量按顺序初始化,递归调用超类的构造方法 子类成员变量按顺序初始化,子类构造方法调用 本文重点演示第三步到第五步: Grandpa类 1 package com.xinye.test; 2 3 public class Grandpa { 4 { 5 System.out.println("执行Grandpa的普通块"); 6 } 7 static

Java对象创建的过程及对象的内存布局与访问定位

这里以HotSpot为例,且所说的对象指普通的Java对象,不包括数组和Class对象等. 1.对象创建的过程 类加载.解析.初始化:虚拟机遇到new时先检查此指令的参数是否能在常量池中找到类的符号引用,并检查符号引用代表的类是否被加载.解析.初始化,若没有则先进行类加载. 对象内存分配:类加载检查通过后,虚拟机为新生对象分配内存,对象所需内存大小在类加载完成后便可完全确定.分配内存的任务等同于从堆中分出一块确定大小的内存. 根据内存是否规整(即用的放一边,空闲的放另一边,是否如此与所使用的垃圾

【深入理解Java虚拟机】Java内存区域模型、对象创建过程、常见OOM

本文内容来源于<深入理解Java虚拟机>一书,非常推荐大家去看一下这本书.最近开始看这本书,打算再开一个相关系列,来总结一下这本书中的重要知识点.呃呃呃,说好的那个图片请求框架呢~  不要急哈,因为这个请求框架设计的内容还是比较广的,目前业余时间正在编写当中,弄好了之后就会放上来.在完成之前,咱还是先来学习一下其他知识. 1.内存模型 java虚拟机在执行java程序的过程中会把它说管理的内存划分为若干个不同的数据区域,如下图所示: 图片来源于网络 (1)程序计数器(Program Count

读《jvm虚拟机》- 对象创建过程

java是面向对象的语言,与对象息息相关, 无时无刻都在创建对象, 对于创建对象,最熟悉的就是new 一个对象出来. Object obj = new Object(); 那么这个new 的过程在虚拟机之中又是怎么实现的呢? 通过阅读<JVM虚拟机>2.3.1对象的创建,简单总结了一下对象创建的过程. 个人见解: 1.虚拟机接收new指令 2.虚拟机接收new指令,检查类加载情况: 当虚拟机遇到一个new指令的时候,首先会去检查该指令中的参数,看是否能在常量池中定位到一个类的符号引用. ps: