Java 中关键字 super 表示的真正对象
java 中的 super,大家都知道是表示一个父类的引用。上次群里见到一个网友询问 super.getClass().getName()的输出问题,大部分都知道输出的是当前这个类的类名。而不是父类的名称。关于这个问题的解释很多,基本都是说 getClass()是一个 final 方法,说这个方法都是调用超父类 Object 的方法。这个解释很好,也容易理解,不过,我们从 super 这个关键词的本质入手,就能更清楚,为什么 super.getClass().getName()会输出当前类的名称了。
先定义两个类,一个父类,一个继承的子类。
父类:
[java] view plain copy
- public class Parent {
-
public String name;
-
private int code;
-
public Parent parent;
-
//定义几个父类成员属性,parent将指向this父类对象
-
public Parent()
-
{
-
//parent指向this,就是当前实例的父类对象,并输出hashcode和给code属性赋值
-
parent=this;
-
code=1001;
-
System.out.println("Parent's code is:"+code);
-
System.out.println("Parent hashcode is:"+this.hashCode());
-
}
-
//定义一个重载的构造方法,用于测试使用了哪个父类构造方法
-
public Parent(String name){
-
parent=this;
-
this.name=name;
-
System.out.println("Parent's name is"+name);
-
code=1002;
-
System.out.println("Parent's code is"+code);
-
System.out.println("Parent hashcode is:"+this.hashCode());
-
}
-
public int getCode()
-
{
-
//打印父类code属性值,
-
System.out.println("Parent :print code :"+code);
-
return code;
-
}
- }
子类:
[java] view plain copy
- public class Child extends Parent{
-
public String childName;
-
private int childCode;
-
//定义两个类的不同修饰符的属性
-
public Child(String childName)
-
{
-
//给属性赋值
-
this.childName=childName;
-
//输出属性值,和类的hashcode值
-
System.out.println("child's childName is:"+childName);
-
System.out.println("child hashcode is:"+this.hashCode());
-
}
-
//测试方法
-
public void test()
-
{
-
//通过super获取父类中的parent,这个成员就表示了父类对象。
-
Child testChild=(Child)super.parent;
-
//强转parent为子类类型。并输出子类中定义的属性,和获取父类getCode()方法
-
System.out.println("testChild name is:"+testChild.childName);
-
testChild.getCode();
-
}
-
public static void main(String[] args) {
-
//实例化
-
Child c=new Child("window");
-
c.test();
-
}
- }
运行输出结果:
Parent's code is:1001
//首先输出这句,可以表示,父类默认的构造方法执行了。
Parent hashcode is:11985823
//这里输出的父类 this 对象的 hashcode 码
child's childName is:window
//调用子类的构造方法,输出childName和hashcode码值,大家可以看到,父类的this对象和子类实例的对象,hashcode码是相同的。
child hashcode is:11985823
//这里执行test()方法的结果。
//使用Child testChild=(Child)super.parent 强制转换父类中指向父类this对象的parent对象为子类Child对象,并输出子类childName属性值。
testChild name is:window
//输出结果上可以看出,这个parent其实就是子类的实例对象。应该说内存中的对象是同一个,只是不同的表示方式。
//执行testChild.getCode(); 调用父类的getCode()方法,输出的和父类构造方法中的code值是相同的。
Parent :print code :1001
从上面可以看出 super 这个关键字所表示的引用父类对象,其实和当前实例化的子类对象在内存中是同一个对象,虽然不知道 sun 的虚拟机是如何实现这个关键字的,但是这个 super 的等同效果就是(Parent)this,这里的 this 是当前代表当前实例化的子类对象。
通过在父类中添加一个 Parent 类型的成员,来指向父类实例化的那个 this 对象,达到引用父类对象的目的,使用 super.parent 来获得父类 parent 对象的引用。
运行的结果表明,这个 parent 应用的其实就是当前子类实例对象,通过强制转换为子类类型,这个父类 Parent 类型的 parent 也可以读取成员 childName 属性值。就表明了他们在内存中是同一个对象。而且他们的 hashcode 值也是相同的。
java 说明中指出,super 是代表父类对象的引用,而 super()表示应用父类的默认构造方法,看起来这个 super 和 c 中的 define 有点相同作用的效果。在这个例子中,super 代表的父类对象的引用,和父类中 parent 代表的都是父类对象的引用,parent 其实就是当前内存中子类对象的引用,如同(Parent)this 一样,这样就可以解释为什么使用(Child)super.parent 可以得到子类成员 childName。
回到开始的问题,super.getClass().getName(),这样,就很容易解释他为什么是输出当前类的名称了,因为构造一个实例化对象,其中,在父类和子类构造方法中,引用的对象都是同一个,都是当前实例化的子类对象。super 关键字应该充当了一个类型转化的作用。
熟悉 c 的知道,c 中经常使用强制转换指针类型来引用一些结构或变量的部分数据,如通过强制转换为不同结构类型,来引用不同数据大小的结构体。这里的应用效果应该等同,通过(Parent)Child 可以通过转换为父类类型而只引用父类类型的那一部分数据。
在 java 中 new 一个对象,和 c 或 c++ 这些和内存打交道的语言一样,都是会分配内存,在 c 中可能更直观一点,我们在此不讨论到底是分配了多大的内存问题。
在 new 一个子类的时候,上面例子看到,父类的成员同样也有赋值初始化,说明,同样在内存中也有保存父类的信息空间,(Object 类的不讨论)。
欢迎来到这里!
我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。
注册 关于