• Java提高篇—对象克隆(复制)

    发布:51Code 时间: 2018-04-08 10:04

  • 假如说你想复制一个简单变量。很简单: int apples = 5; int pears = apples; 不仅仅是int类型,其它七种原始数据类型(boolean,char,byte,short,float,double.long)同样适用于该类情况。 但是如果你复制的...

  • 假如说你想复制一个简单变量。很简单:

    int apples = 5;    

    int pears = apples; 

    不仅仅是int类型,其它七种原始数据类型(boolean,char,byte,short,float,double.long)同样适用于该类情况。

    但是如果你复制的是一个对象,情况就有些复杂了。

    假设说我是一个beginner,我会这样写:

    class Student {    

       private int number;     

       public int getNumber() {  

           return number;    

       }  

       

         public void setNumber(int number) {    

             this.number = number;    

         }   

      }    

      public class Test {    

         public static void main(String args[]) {   

             Student stu1 = new Student();    

             stu1.setNumber(12345);    

             Student stu2 = stu1;             

       

             System.out.println("学生1:" + stu1.getNumber());    

             System.out.println("学生2:" + stu2.getNumber());   

         }    

      } 

    结果:

    学生1:12345   

    学生2:12345 

    这里我们自定义了一个学生类,该类只有一个number字段。

    我们新建了一个学生实例,然后将该值赋值给stu2实例。(Student stu2 = stu1;)

    再看看打印结果,作为一个新手,拍了拍胸腹,对象复制不过如此!

    难道真的是这样吗?

    我们试着改变stu2实例的number字段,再打印结果看看:

    stu2.setNumber(54321);     

    System.out.println("学生1:" + stu1.getNumber());    

    System.out.println("学生2:" + stu2.getNumber()); 

    结果:

    学生1:54321    

    学生2:54321

    这就怪了,为什么改变学生2的学号,学生1的学号也发生了变化呢?

    原因出在(stu2 = stu1) 这一句。该语句的作用是将stu1的引用赋值给stu2,这样,stu1和stu2指向内存堆中同一个对象。如图:

    那么,怎样才能达到复制一个对象呢?

    是否记得万类之王Object。它有11个方法,有两个protected的方法,其中一个为clone方法。

    在Java中所有的类都是缺省的继承自Java语言包中的Object类的,查看它的源码,你可以把你的JDK目录下的src.zip复制到其他地方然后解压,里面就是所有的源码。发现里面有一个访问限定符为protected的方法clone():

    /*  

    Creates and returns a copy of this object. The precise meaning of "copy" may depend on the class of the object.  

    The general intent is that, for any object x, the expression:  

    1) x.clone() != x will be true  

    2) x.clone().getClass() == x.getClass() will be true, but these are not absolute requirements.  

    3) x.clone().equals(x) will be true, this is not an absolute requirement.  

    */  

    protected native Object clone() throws CloneNotSupportedException; 

    仔细一看,它还是一个native方法,大家都知道native方法是非Java语言实现的代码,供Java程序调用的,因为Java程序是运行在JVM虚拟机上面的,要想访问到比较底层的与操作系统相关的就没办法了,只能由靠近操作系统的语言来实现。

    第一次声明保证克隆对象将有单独的内存地址分配。

    第二次声明表明,原始和克隆的对象应该具有相同的类类型,但它不是强制性的。

    第三声明表明,原始和克隆的对象应该是平等的equals()方法使用,但它不是强制性的。

    因为每个类直接或间接的父类都是Object,因此它们都含有clone()方法,但是因为该方法是protected,所以都不能在类外进行访问。

    要想对一个对象进行复制,就需要对clone方法覆盖。

    本文原作者:萌小Q
    文章来源:Java团长
  • 上一篇:2018年Java,Web和移动程序员学习的12个框架

    下一篇:Java工程师的职业规划

网站导航
Copyright(C)51Code软件开发网 2003-2018 , 沪ICP备16012939号-1