java中,super.clone()在存在继承关系的类上,最终克隆的是哪个类?

clone在克隆存在继承关系的类时,super.clone();最终克隆的是哪个类的问题:

当子类和父类都重写了clone()方法,因为clone必须最终调用object类的clone方法才能完成克隆,那么,对子类对象克隆时,由于子类的clone方法里一定存在super.clone(),那么就会调用父类的clone方法,最终返回的为什么不是父类的克隆对象?在这过程中,为什么没发生父类对象赋给子类引用时产生的编译错误?

示例代码:

class Father implements Cloneable{
    public Father clone() throws CloneNotSupportedException {
        System.out.println(super.clone().getClass()+"--Father类clone方法执行了");
        return (Father)super.clone();
    }
}
class Son extends Father implements Cloneable{
    int i=9;//是Test3类中独有的变量,用来测试clone是否成功

    public Son clone() throws CloneNotSupportedException {
        System.out.println(super.clone().getClass()+"--Son类clone方法执行了");
        return (Son)super.clone();
    }
}
class Test{
    public static void main(String[] args) throws CloneNotSupportedException {
        Son test=new Son().clone();
        System.out.println(test.i);
        //程序正常运行,最后输出i的值9
        //为什么?
        //我以为会发生:new Son().clone()→
        //触发return (Son)super.clone();,找到super代表的父类Father
        // 调用Father 的 clone()方法,经过强制类型转化,得到Father类型引用变量→
        //将收到的Father类型引用变量,强制类型转化:(Son),
        //由于Son是Father子类,Son中·存在Father中没有的变量i,所以这种转换会报错
        //为什么结果却能正确执行?
    }
}

运行结果:

class com.cskaoyan.objectapi.clone.Son--Father类clone方法执行了
class com.cskaoyan.objectapi.clone.Son--Son类clone方法执行了
class com.cskaoyan.objectapi.clone.Son--Father类clone方法执行了
9

可见,子类clone方法调用super.clone()的确会执行父类clone方法,但父类中调用super.clone(),也就是调用
object类中的clone方法时,产生的一定也是最开始调用super.clone()那个类的对象,也就是子类对象,事实上,
无论存在多少层继承关系,在层层super.clone()的过程中,每一层返回的对象都是最开始调用clone方法的那个对象,

原因在于,其实每个对象中都存储了一些额外信息,其中就包括,存储了对象所属的实际类型,所以Clone方法访问对象的这个信息就能知道对象的实际类型,从而按照对象的实际类型复制对象

这种情况产生原因是,clone方法是native方法,底层是由c/c++实现的,所以不受java语法约束,可视为一种语法糖,
这是一种开后门的语法,就像包装类可以使用运算符直接运算一样无耻又方便
同理,另一个native方法:Thread类中的sleep()方法也能实现这种语法糖效果,即无论是哪个线程对象调用的它,
其结果都是让执行这句代码所在的线程休眠(而非调用sleep方法的线程对象代表的那个线程)

  • 3
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值