黑马程序员技术交流社区
标题: 构造代码块和显示初始化的顺序问题 [打印本页]
作者: 高原 时间: 2012-7-7 17:23
标题: 构造代码块和显示初始化的顺序问题
本帖最后由 高原 于 2012-7-7 23:30 编辑
毕老师第六天的课程中讲到对象的初始化过程。他说:先对属性显示初始化,然后再进行构造代码块初始化。
我写了个程序,与老师说的不太一样。- public class Person {
-
- {
- System.out.println("name= " + this.name);
- }
- public String name = "zhangsan";
-
- static {
- Person.country = "static country";
- System.out.println("country= " + Person.country);
- }
- public static String country = "cn";
- public Person(String name) {
- this.name = name;
- }
- }
- public class TestInit {
- public static void main(String[] args) {
- Person p = new Person("lisi");
- System.out.println("name= " + p.name);
- System.out.println("country= " + Person.country);
- }
- }
复制代码 打印结果为:
country= static country
name= null
name= lisi
country= cn
这说明构造代码块先执行了,然后显示初始化才执行。
并且,静态代码块的执行时机也比类变量的显示初始化要早。
所以构造代码块,显示初始化谁在代码中比较靠前,谁就先执行,静态代码块和类变量显示初始化也是同样的。
我觉得对象初始化过程是这样的:
1. 首先加载类到内存中
2. 对于类变量,系统默认初始化为null或0等默认值
3. 执行静态代码块或类变量的显示初始化(代码中谁在前面谁就先执行)
4. 在堆内存中开辟空间,分配内存地址
5. 对于成员变量,首先系统默认初始化为null或者0等默认值
6. 然后执行构造代码块或成员变量显示初始化(代码中谁在前面谁就先执行)
7. 最后才是构造方法初始化
8. 将堆内存中的地址付给栈内存中的引用
大家怎样认为?
作者: 邵阳 时间: 2012-7-7 17:42
本帖最后由 邵阳 于 2012-7-7 20:50 编辑
我不认为楼主是正确的,恰恰相反我认为楼主的代码正好印证了毕老师的说法,我慢慢解释。对象的初始化过程如下:
1:因为new用到了Person.class,所以会先找到Person.class文件并加载到内存中。
2:执行该类型中的static代码块,如果有的话,给Person.class类进行初始化。
3:在对内存中开辟空间,分配内存地址。
4:在堆内存中建立对象的特有属性,并进行默认初始化。
5:对属性显示初始化。
6:对对象进行构造代码块初始化。
7:对对象进行相应的构造函数初始化。
8:将内存地址付给栈内存正的p变量。
public class Person {
{
System.out.println("name= " + this.name);//2:建立对象,默认初始化为null
}
public String name = "zhangsan";
static { // 1:首先执行static代码块里面的内容,打印结果是country= static country
Person.country = "static country";
System.out.println("country= " + Person.country);
}
public static String country = "cn";//4:虽然对象一加载内存,都已经初始化。但是其
//System.out.println("country= " + Person.country);放在最后,所以最后执行。
public Person(String name) { //3:构造函数初始化。
this.name = name;
}
}
public class TestInit {
public static void main(String[] args) {
Person p = new Person("lisi");
System.out.println("name= " + p.name);
System.out.println("country= " + Person.country);
}
}
1:country= static country
2:name= null
3:name= lisi
4:country= cn
解释的有点牵强,我尽力啦,以后还会继续补充。
我现在搞懂啦,毕老师少说了一条,因为他讲的时候,初始变量在构造代码块的上面。
7楼同学说的很对
6. 执行构造代码块或成员变量显示初始化(代码中谁在前面谁就先执行)
所以很佩服楼主的挑战权威的精神,
强烈版主建议给楼主加3分,七楼同学加2分
作者: 康大玮 时间: 2012-7-7 18:04
以前有人回答过这个问题 我中介了一下 :
1,静态代码块、构造代码块和构造函数的区别
静态代码块:用于给类初始化,类加载时就会被加载执行,只加载一次。
构造代码块:用于给对象初始化的。只要建立对象该部分就会被执行,且优先于构造函数。
构造函数: 给对应对象初始化的,建立对象时,选择相应的构造函数初始化对象。
创建对象时,三者被加载执行顺序:静态代码块--->构造代码块--->构造函数
2,什么时候会加载类?
使用到类中的内容时加载:有三种情况
1.创建对象:new StaticCode();
2.使用类中的静态成员:StaticCode.num=9; StaticCode.show();
3.在命令行中运行:java StaticCodeDemo
3,类所有内容加载顺序和内存中的存放位置:
利用语句进行分析。
1.Person p=new Person("zhangsan",20);
该句话所做的事情:
1.在栈内存中,开辟main函数的空间,建立main函数的变量 p。
2.加载类文件:因为new要用到Person.class,所以要先从硬盘中找到Person.class类文件,并加载到内存中。
加载类文件时,除了非静态成员变量(对象的特有属性)不会被加载,其它的都会被加载。
记住:加载,是将类文件中的一行行内容存放到了内存当中,并不会执行任何语句。---->加载时期,即使有输出语句也不会执行。
静态成员变量(类变量) ----->方法区的静态部分
静态方法 ----->方法区的静态部分
非静态方法(包括构造函数) ----->方法区的非静态部分
静态代码块 ----->方法区的静态部分
构造代码块 ----->方法区的静态部分
注意:在Person.class文件加载时,静态方法和非静态方法都会加载到方法区中,只不过要调用到非静态方法时需要先实例化一个对象
,对象才能调用非静态方法。如果让类中所有的非静态方法都随着对象的实例化而建立一次,那么会大量消耗内存资源,
所以才会让所有对象共享这些非静态方法,然后用this关键字指向调用非静态方法的对象。
3.执行类中的静态代码块:如果有的话,对Person.class类进行初始化。
4.开辟空间:在堆内存中开辟空间,分配内存地址。
5.默认初始化:在堆内存中建立 对象的特有属性,并进行默认初始化。
6.显示初始化:对属性进行显示初始化。
7.构造代码块:执行类中的构造代码块,对对象进行构造代码块初始化。
8.构造函数初始化:对对象进行对应的构造函数初始化。
9.将内存地址赋值给栈内存中的变量p。
2.p.setName("lisi");
1.在栈内存中开辟setName方法的空间,里面有:对象的引用this,临时变量name
2.将p的值赋值给this,this就指向了堆中调用该方法的对象。
3.将"lisi" 赋值给临时变量name。
4.将临时变量的值赋值给this的name。
3.Person.showCountry();
1.在栈内存中,开辟showCountry()方法的空间,里面有:类名的引用Person。
2.Person指向方法区中Person类的静态方法区的地址。
3.调用静态方法区中的country,并输出。
注意:要想使用类中的成员,必须调用。通过什么调用?有:类名、this、super
作者: 温少邦 时间: 2012-7-7 18:27
邵阳 发表于 2012-7-7 17:42 
我不认为楼主是正确的,恰恰相反我认为楼主的代码正好印证了毕老师的说法,我慢慢解释。对象的初始化过程如 ...
我也有点疑问
System.out.println("name= " + this.name);
这是在构造代码块里面的
为什么显示null?
如果属性的显式初始化在构造代码块之前
那么 public String name = "zhangsan"; 先执行
这个name属性现在应该是"zhangsan"了
属性是初始化了 但是没有显式赋的初值
这是不是对"属性的显式初始化"理解的问题?
作者: 邵阳 时间: 2012-7-7 18:46
温少邦 发表于 2012-7-7 18:27 
我也有点疑问
System.out.println("name= " + this.name);
这里我也没有理解很深刻,不过我死记硬背记着啦。我啥时候想明白了继续补充
作者: 黎健东 时间: 2012-7-7 19:35
温少邦 发表于 2012-7-7 18:27 
我也有点疑问
System.out.println("name= " + this.name);
3楼说的很清楚了
我简单说一下我的理解简化后的执行过程
1、new一个对象的时候,就开始加载静态部分了,先执行静态方法static{}、静态static String country属性部分,然后执行静态代码块{System.out.println(("name= " + this.name);}部分
2、new Person("lisi")这里,调用构造函数,Person(String name),这个时候就 p这个对象变量的name属性就被赋值了。
3、然后依次执行main函数中打印部分,就依次输出控制台显示的结果了
作者: 曹恒业 时间: 2012-7-7 20:23
本帖最后由 曹恒业 于 2012-7-7 20:38 编辑
- public class Person {
-
- {
- System.out.println("name= " + this.name);
- }
- public String name = "zhangsan"; //如果将这个对name的显示初始化放到构造代码块之前,那么就符合了毕老师视频里的结论。最后name打印的结果就是“zhangsan”。若有怀疑,大家可以试试,亲自动手,会印象更深刻。
-
- static {
- Person.country = "static country";
- System.out.println("country= " + Person.country);
- }
- public static String country = "cn";
- public Person(String name) {
- this.name = name;
- }
- }
- public class TestInit {
- public static void main(String[] args) {
- Person p = new Person("lisi");
- System.out.println("name= " + p.name);
- System.out.println("country= " + Person.country);
- }
- }
复制代码 首先,感谢楼主为我们发现了这么一个容易让大家忽视的细节,确实,一开始我也不明白结果为何和毕老师说的不同,但我最终还是决定相信JVM。因此自己写代码试验了良久,正好把全部关于初始化的顺序都做了实验。终于确定了LZ的猜想。
6. 执行构造代码块或成员变量显示初始化(代码中谁在前面谁就先执行) 这里的详细解释见代码中注释。
作者: 邵阳 时间: 2012-7-7 20:31
曹恒业 发表于 2012-7-7 20:23 
首先,感谢楼主为我们发现了这么一个容易让大家忽视的细节,确实,一开始我也不明白结果为何和毕老师说的不 ...
你的很对啊
作者: 高原 时间: 2012-7-7 23:08
邵阳 发表于 2012-7-7 17:42 
我不认为楼主是正确的,恰恰相反我认为楼主的代码正好印证了毕老师的说法,我慢慢解释。对象的初始化过程如 ...
哥们你太帅了,我感动了,谢谢啊!!!
希望斑竹能加分啊
作者: 高原 时间: 2012-7-7 23:10
曹恒业 发表于 2012-7-7 20:23 
首先,感谢楼主为我们发现了这么一个容易让大家忽视的细节,确实,一开始我也不明白结果为何和毕老师说的不 ...
哈哈,过奖了,我也是一时好奇才做了这个实验~:)
欢迎光临 黑马程序员技术交流社区 (http://bbs.itheima.com/) |
黑马程序员IT技术论坛 X3.2 |