一、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“关键字,就可以使用父类中的同名成员变量了。

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

  1. java对象创建过程

    1.jvm找到class文件路径. 2.jvm载入class文件,静态初始化,创建一个class对象. 3.为即将创建的对象分配内存空间. 4.对分配的空间进行清零,例如:int清除为0,boolea ...

  2. java对象创建过程简介

    这是看书的记录,字有点丑啊还是将就搬上来 -.-,等把后面看了完善图

  3. Java中对象创建过程

    本文介绍的对象创建过程仅限于普通Java对象,不包括数组和Class对象. 1.类加载检查 虚拟机遇到一条new指令时,首先去检查该指令的参数能否在常量池中定位到一个类的符号引用,并且检查这个符号引用 ...

  4. 深入学习Java对象创建的过程:类的初始化与实例化

    在Java中,一个对象在可以被使用之前必须要被正确地初始化,这一点是Java规范规定的.在实例化一个对象时,JVM首先会检查相关类型是否已经加载并初始化,如果没有,则JVM立即进行加载并调用类构造器完 ...

  5. Java中类加载过程和对象创建过程

    类加载过程: 1, JVM会先去方法区中找有没有相应类的.class存在.如果有,就直接使用:如果没有,则把相关类的.class加载到方法区 2, 在.class加载到方法区时,会分为两部分加载:先加 ...

  6. java基础---类加载和对象创建过程

    类中可以存在的成员: class A{ 静态成员变量: 非静态成员变量: 静态函数: 非静态函数: 构造函数 A(..){...} 静态代码块 static{...} 构造代码块 {...} } 类加 ...

  7. java类加载、对象创建过程

    类加载过程: 1, JVM会先去方法区中找有没有相应类的.class存在.如果有,就直接使用:如果没有,则把相关类的.class加载到方法区 2, 在.class加载到方法区时,会分为两部分加载:先加 ...

  8. COM对象创建过程

    在客户端需要调用COM组件时,通常调用windowAPI函数: STDAPI CoCreateInstance( REFCLSID rclsid, //创建的Com对象的类标识符(CLSID) LPU ...

  9. Java内存区域与对象创建过程

    一.java内存区域 Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域.这些区域都有各自的用途,以及创建和销毁的时间,有的区域随着虚拟机进程的启动而存在,有的区域则 ...

随机推荐

  1. net core mysql 组件记录

    nuget 下 Pomelo.EntityFrameworkCore.MySql (据说是微软官方推荐) MySql.Data.EntityFrameworkCore (甲骨文出品) 使用方式等同于  ...

  2. 一图看懂JVM,JRE,JDK的关系

  3. Spring框架中用到的设计模式(转)

    主要参考这篇文章 http://blog.didispace.com/spring-design-partern/

  4. CTF-安恒19年一月月赛部分writeup

    CTF-安恒19年一月月赛部分writeup MISC1-赢战2019 是一道图片隐写题 linux下可以正常打开图片,首先到binwalk分析一下. 里面有东西,foremost分离一下 有一张二维 ...

  5. 20155226 mini DC 课堂测试补交

    由于电脑突然出了点问题,我没有完成mini DC这个测试,现将测试内容及结果补交 题目如下 提交测试截图和码云练习项目链接,实现Linux下dc的功能,计算后缀表达式的值 代码如下 MyDC.clas ...

  6. 20155232 2016-2017-3 《Java程序设计》第3周学习总结

    20155232 2016-2017-3 <Java程序设计>第3周学习总结 教材学习内容总结 第四章 认识对象 1.对象(Object):存在的具体实体,具有明确的状态和行为. 2.类( ...

  7. 20155321 《Java程序设计》实验五 网络编程与安全

    实验内容 两人一组结对编程: 参考http://www.cnblogs.com/rocedu/p/6766748.html#SECDSA 结对实现中缀表达式转后缀表达式的功能 MyBC.java 结对 ...

  8. 从原理到代码:大牛教你如何用 TensorFlow 亲手搭建一套图像识别模块 | AI 研习社

    从原理到代码:大牛教你如何用 TensorFlow 亲手搭建一套图像识别模块 | AI 研习社 PPT链接: https://pan.baidu.com/s/1i5Jrr1N 视频链接: https: ...

  9. (ex)Lucas总结

    (ex)Lucas总结 普通Lucas 求 \[ C_n^m\;mod\;p \] 其中\(n,m,p\leq 10^5\)其中\(p\)为质数 公式不难背,那就直接背吧... \[ C_n^m\;m ...

  10. 一个奇怪的JS函数

    今天在分析一个jQuery插件源码的时候,发现了一个奇怪的函数. 这个函数的目的是为数字补零,如传入7,输出07,传入12输出12.由于是对时间补零,只截取后两位. // add leading ze ...