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

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

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

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

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

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

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

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

  1. public class JavaTest
  2. {
  3. //成员变量
  4. int count=20;
  5.  
  6. //非静态初始化代码块
  7. {
  8. count = 12;
  9. System.out.print("占个位置而已");
  10. }
  11.  
  12. //构造器
  13. public JavaTest()
  14. {
  15. System.out.println(count);
  16. }
  17.  
  18. public JavaTest(String name)
  19. {
  20. System.out.println(count);
  21. }
  22. }

编译后等价于:

  1. public class JavaTest
  2. {
  3. //成员变量
  4. int count;
  5.  
  6. //非静态初始化代码块
  7. {
  8. System.out.print("占个位置而已");
  9. }
  10.  
  11. //构造器
  12. public JavaTest()
  13. {
  14. count=20;//原来的成员变量赋值
  15. count=12;//原来的初始化代码块赋值
  16. System.out.println(count);
  17. }
  18.  
  19. public JavaTest(String name)
  20. {
  21. count=20;//原来的成员变量赋值
  22. count=12;//原来的初始化代码块赋值
  23. System.out.println(count);
  24. }
  25. }

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

  1. class base
  2. {
  3. private int i = 2;
  4.  
  5. public Base()
  6. {
  7. 这里是1
  1. this.display();
  2. }
  3.  
  4. public void display()
  5. {
  6. System.out.println(i);
  7. }
  8. }
  9.  
  10. class Derived extends base
  11. {
  12. private int i = 22;
  13.  
  14. public Derived()
  15. {
  16. i = 222;
  17. }
  18.  
  19. public void display()
  20. {
  21. System.out.println(i);
  22. }
  23. }
  24.  
  25. public class Test
  26. {
  27. public static void main(String[] args)
  28. {
  29. new Derived();
  30. }
  31. }

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

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

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

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

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

  1. class base
  2. {
  3. private int i = 2;
  4.  
  5. public Base()
  6. {
  7. this.display();
  8. }
  9.  
  10. public void display()
  11. {
  12. System.out.println("我是父类"+i);
  13. }
  14. }
  15.  
  16. class Derived extends base
  17. {
  18. private int i = 22;
  19.  
  20. public Derived()
  21. {
  22. i = 222;
  23. }
  24.  
  25. public void display()
  26. {
  27. System.out.println("我是子类"+i);
  28. }
  29. }

情况一:

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. centos 安装 telnet

    (转)centos7安装telnet服务 场景:在进行Telnet测试时候,发现无法连接,所以还得把这个软件也安装了 1 CentOS7.0 telnet-server 启动的问题 解决方法:   先 ...

  2. Python3 透明网桥算法

    import time #定义网桥1 b1 = {} port_list1 = [1, 2] #主机列表 L1 = ['a','b','c'] L2 = ['d','e'] L = [L1,L2] d ...

  3. 微信小程序判断按钮是否显示,或者隐藏

    js中: onLoad:function(options) { this.setData({ orderstate:'待送检' }) },   WXML中: <view wx:if=" ...

  4. day 3 局部变量 全局变量

    1.局部变量 2.全局变量(死歌的大招)函数前面声明的都是全局变量 3.全局变量和局部变量的区别 1)老方法 def get_temper(): temper = 33 return temper d ...

  5. SRM First Problem && SRM 638 250pts NamingConvention

    NamingConvention 题意: 给一个字符串,删掉所有的'_',然后将‘_'后的第一个字符改成大写. 代码: #include<bits/stdc++.h> using name ...

  6. 前端 CDNJS 库及 Google Fonts、Ajax 和 Gravatar 国内加速服务

    由于某些众所周知的原因,好多开源的 JS 库采用的国外 CDN 托管方式在国内访问速度不如人意.所以我们特意制作了这个公益项目,托管了 CDNJS 的所有开源 JS 库以及反代了 Google Fon ...

  7. shell 参数

    转:http://hi.baidu.com/ipvsadm/item/489d9e16460195ddbe9042ee linux中shell变量$#,$@,$0,$1,$2的含义解释 linux中s ...

  8. 【SpringCloud】第十二篇: 断路器监控(Hystrix Turbine)

    前言: 必需学会SpringBoot基础知识 简介: spring cloud 为开发人员提供了快速构建分布式系统的一些工具,包括配置管理.服务发现.断路器.路由.微代理.事件总线.全局锁.决策竞选. ...

  9. i3wm随笔 1

    快捷键 mod+0 退出 mod+v 垂直分割 mod+h 水平风格

  10. 转载-找圆算法((HoughCircles)总结与优化-霍夫变换

    原文链接: http://www.opencv.org.cn/forum.php?mod=viewthread&tid=34096   找圆算法((HoughCircles)总结与优化 Ope ...