静态变量、实例变量初始化时机,以及子类隐藏父类成员时,创建子类对象的一些问题
程序员文章站
2022-05-21 22:45:21
...
最近在复习java基础,类加载的机制,看到很多篇好的博客,但是比较分散,所以自己总结了一下
代码:
package acx;
public class TestClass {
public String count="testclass";//实例变量
public static TestClass testClass=getInstence();
public static int count1;
public static int count2=0;
private TestClass() {//父类构造方法
System.out.println(out());
count1=count1+1;
count2=count2+1;
System.out.println(out());
}
public static TestClass getInstence(){//单例模式 可以了解一下
if(testClass==null){
return new TestClass();
}else{
return testClass;
}
}
public String out(){
return (""+count+" "+count1+" "+count2+" "+"constructor");
}
static class Sub extends TestClass{//静态内部类
public String count="testsub";
public String out(){
return (""+count+" "+count1+" "+count2+" "+"SUB");
}
}
public static void main(String[] args) {
Sub sub=new Sub();
System.out.println(sub.out());
}
}
运行结果截图:
为什么是 这样 呢?
以下是个人理解:
main方法里创建子类实例,但是此时父类还未创建,所以先创建父类(新手用不来这个编辑器,怎么选择不了,好累)
创建父类过程:{
在类加载的准备阶段,为静态变量分配内存并赋默认值此时 count1=0,count2=0,testClass=null
初始化阶段为静态变量赋值、执行静态代码块,为testClass赋值调用构造方法产生类的对象,此时为实例变量在堆里分配内存并赋值count=“testClass”,接着执行输出语句,输出第一行。
然后count+1,count2+1.输出第二行。
继续为静态变量赋值,count1没有操作,仍为1,count2被重新赋值为0
}
创建父类之后初始化子类,子类没有声明静态变量,可能会有疑问,没有继承父类的静态变量吗?没有,等下解释
执行子类的默认无参构造方法 注意(构造方法中有 一句 super();),但是子类的count已经覆盖了父类的count属性,但是还未初始化,所以此时输出null 1 0 SUB第三行,因为此时运行的大环境是在子类的无参构造方法里,this指的其实是子类,所以调用的out方法为子类的out方法
接着count1+1,count2+1,此时count1=2,count2=1;同理接着输出第四行
到此为止子类构造方法里的supper()执行完成,子类实例变量初始化,子类count=testsub
最后的输出语句:子类对象调用out方法输出第五行
补充 (这才是重要的):static 变量是类变量,不属于任何对象,但是子类对象可以访问父类的静态变量,如果子类也声明了同名的变量也不会覆盖,只会访问子类自己的静态变量,因为用子类名和父类名访问的是不同的内存空间。
还有构造方法和实例变量初始化究竟是怎样的执行顺序(个人理解):这两个的执行在JVM中都归结到 <init> 方法,在init中的顺序是:supper();+ 实例变量初始化+构造方法里面的代码
欢迎批评指正