在创建子类对象时,会把父类里的成员变量和方法也加载进内存(因为要加载进内存,所以要看下这些数据是怎么初始化的,所以调用了父类的构造,仅此而已,并不是去创建了父类对象)
然后用this和super这两个引用来区分是父类的还是子类的,但是这个内存区域是子类的内存区域,绝不是父类的 。this
指向了不仅父类中可继承的成员变量和可继承的方法外,它还指向了子类的成员变量和方法 而super
仅仅只是指向了子类对象中从父类继承的成员变量和方法。
也就是说,在子类对象里(堆空间),不仅有自己的属性方法还有一块空间是指向父类的属性方法;
在字节码中子类会有个u2类型的父类索引,属于CONSTANT_Class_info类型,通过CONSTANT_Class_info的描述可以找到CONSTANT_Utf8_info,然后可以找到指定的父类啊啥的。
class A
{
int name = 10;
}
class B extends A
{
int name = 5;
public B(){
super();
this.name = 101; //这个修改的是子类中的name,现在子类中的name是子类独有的属性。
super.name = 100; //这个是否修改了父类的name属性的值?
//他修改的只是子类对象内存中引用的父类的属性的值,跟父类本身没有关系,父类也没有创建对象
//子类对象中有一块是引用父类的
}
}
class Test1
{
public static void main(String[] args)
{
A a = new B();
System.out.println(a.name); //静态绑定,看左边,所以调用的是A的name属性100
//静态绑定 动态绑定:https://blog.youkuaiyun.com/BigDevil_/article/details/103295593
}
}
class A
{
int name = 10;
}
class B extends A
{
public B(){
super();
this.name = 101; //这个时候。因为B里没有name属性,
//所以调用的是他内存中父类的name属性,修改的也是父类的属性
super.name = 100; //super同样也是修改的父类的属性
//注意:跟A类本身属性没关系,只是b对象内存中的父类的属性
}
}
class Test2
{
public static void main(String[] args)
{
A a = new B();
System.out.println(a.name); //100 因为100把101给覆盖了
}
}
this和super
1、this指当前对象,super指继承的父类
this
指向了不仅父类中可继承的成员变量和可继承的方法外,它还指向了子类的成员变量和方法 而super
仅仅只是指向了子类对象中从父类继承的成员变量和方法。
2、this和super不能同时出现在一个构造函数里面
3、this或super调构造器时必须放在第一句,super()调用父类构造方法,this() 调用本类构造方法,及时调用了this,默认还会调用super。不放在第一句编译报错:
Constructor call must be the first statement in a constructor
4、this和super都指的是对象,所以,均不可以在static环境下使用
5、如果父类只有一个带参构造方法,那么子类必须调用父类的构造方法super(… params)。注意:子类每个构造方法中默认有super(),前提是父类有无参构造。
public class Test {
public static void main(String[] args) throws Exception {
B11 b11 = new B11();
}
}
class A11{
public A11(){
System.out.println("A构造器");
}
public A11(int a) {
System.out.println("A有参构造器");
}
}
class B11 extends A11{
public B11(){
this(1);
System.out.println("B构造器");
}
public B11(int a){
super(1);
System.out.println("B有参构造器");
}
}
结果:
A有参构造器
B有参构造器
B构造器