关于 Java 中对象的复制

阅读更多

关于 Java 中对象的复制

        Java 中的赋值操作符与 C++ 中的不一样。在 C++ 中,这条语句:bc2 = bc1;将一个名为 bc1 的对象的所有数据都拷贝到名为 bc2 的对象中。也就是说这条语句执行后,程序中有两个含有相同数据的对象。然而在 Java 中,这条相同的赋值语句只向 bc2 中拷贝了 bc1 指向的存储地址,现在 bc1 和 bc2 实际上指的是同一个对象,它们都是这个对象的引用。这样大大提高了内存使用效率,同时也容易让一些对内存了解不深的朋友带来一些使用上的错误。比如 bc1.add(25);buc2.add(20);执行之后 bc1 增加了 45。而作者本意可能是只是让 bc1 增加 25,bc2 增加 20 而已。这说明了,在 Java 中,bc2 = bc1; 并不是真正意义上的复制。那么在 Java 中如何进行对象复制呢?作者结合 Java 数据结构相关知识,总结了一些项目中的经验,希望可以和大家共同探讨一下这个问题。
        没有使用对象复制的代码:
        银行帐户源代码 BankAccount.java:

view plain copy to clipboard print ?
  1. package clone;  
  2. public class BankAccount{  
  3.     private double balance;  
  4.     public BankAccount(double ini){  
  5.         this.balance = ini;  
  6.     }  
  7.       
  8.     public double getBalance() {  
  9.         return balance;  
  10.     }  
  11.     public void setBalance(double balance) {  
  12.         this.balance = balance;  
  13.     }  
  14.     public void add(double give){  
  15.         this.balance += give;  
  16.     }  
  17.       
  18.     public void redu(double give){  
  19.         this.balance -= give;  
  20.     }  
  21. }  

package clone; public class BankAccount{ private double balance; public BankAccount(double ini){ this.balance = ini; } public double getBalance() { return balance; } public void setBalance(double balance) { this.balance = balance; } public void add(double give){ this.balance += give; } public void redu(double give){ this.balance -= give; } }
        程序入口 TestClone.java:

view plain copy to clipboard print ?
  1. package clone;  
  2. public class TestClone {  
  3.     public static void main(String[] args) {  
  4.         BankAccount bc1 = new BankAccount(1000.0);  
  5.         BankAccount bc2 = bc1;  
  6.         bc1.add(25);  
  7.         bc2.add(20);  
  8.         System.out.println("用户 1 的帐户余额为:" + bc1.getBalance());  
  9.         System.out.println("用户 2 的帐户余额为:" + bc2.getBalance());  
  10.     }  
  11. }  

package clone; public class TestClone { public static void main(String[] args) { BankAccount bc1 = new BankAccount(1000.0); BankAccount bc2 = bc1; bc1.add(25); bc2.add(20); System.out.println("用户 1 的帐户余额为:" + bc1.getBalance()); System.out.println("用户 2 的帐户余额为:" + bc2.getBalance()); } }
        执行 TestClone.java,打印结果如下:
用户 1 的帐户余额为:1045.0
用户 2 的帐户余额为:1045.0

        这下用户 2 要哭了:自己辛辛苦苦攒的一点钱都存别人帐户里边去了。怎么解决这个问题呢?用户 2 不应该使用等号,而应该进行对象复制。那么在 Java 中怎样进行对象复制呢?有两种办法。
        Java 中对象的复制办法一

 

        一开始就创建两个不同的对象,然后分别拷贝每一个字段。注意:等号是不起复制作用的!这里说的拷贝并非用等号进行,而是手工复制。代码说明如下。
        银行帐户源代码不变 BankAccount.java:

view plain copy to clipboard print ?
  1. package clone;  
  2. public class BankAccount{  
  3.     private double balance;  
  4.     public BankAccount(double ini){  
  5.         this.balance = ini;  
  6.     }  
  7.       
  8.     public double getBalance() {  
  9.         return balance;  
  10.     }  
  11.     public void setBalance(double balance) {  
  12.         this.balance = balance;  
  13.     }  
  14.     public void add(double give){  
  15.         this.balance += give;  
  16.     }  
  17.       
  18.     public void redu(double give){  
  19.         this.balance -= give;  
  20.     }  
  21. }  

package clone; public class BankAccount{ private double balance; public BankAccount(double ini){ this.balance = ini; } public double getBalance() { return balance; } public void setBalance(double balance) { this.balance = balance; } public void add(double give){ this.balance += give; } public void redu(double give){ this.balance -= give; } }
        在程序入口 TestClone.java 中进行对象复制:

view plain copy to clipboard print ?
  1. package clone;  
  2. public class TestClone {  
  3.     public static void main(String[] args) {  
  4.         BankAccount bc1 = new BankAccount(1000.0);  
  5.         BankAccount bc2 = new BankAccount(1000.0);  
  6.         bc1.add(25);  
  7.         bc2.add(20);  
  8.         System.out.println("用户 1 的帐户余额为:" + bc1.getBalance());  
  9.         System.out.println("用户 2 的帐户余额为:" + bc2.getBalance());  
  10.     }  
  11. }  

package clone; public class TestClone { public static void main(String[] args) { BankAccount bc1 = new BankAccount(1000.0); BankAccount bc2 = new BankAccount(1000.0); bc1.add(25); bc2.add(20); System.out.println("用户 1 的帐户余额为:" + bc1.getBalance()); System.out.println("用户 2 的帐户余额为:" + bc2.getBalance()); } }
        执行 TestClone.java,打印结果如下:
用户 1 的帐户余额为:1025.0
用户 2 的帐户余额为:1020.0

        用户 2 终于把自己挣的血汗钱存入了自己的帐户里去了。但是 Java 程序员要哭了:这里只是一个简单的例子,如果对象比较复杂,每次都要复制,而且还要处处考虑是不是又进行赋值引用了?这岂不麻烦?为了克服这个问题,Java 引入了克隆的概念。
        Java 中对象的复制办法二

 

        使用克隆进行对象复制。Java API 里解释:java.lang.Object.clone() 可以创建一个当前实例的拷贝。前提是当前实例的对象必须实现 java.lang.Cloneable 接口,然后再重载 java.lang.Object 的 clone 方法。代码说明如下。
        银行帐户源代码 BankAccount.java:

view plain copy to clipboard print ?
  1. package clone;  
  2. public class BankAccount implements Cloneable{  
  3.     private double balance;  
  4.     public BankAccount(double ini){  
  5.         this.balance = ini;  
  6.     }  
  7.       
  8.     protected Object clone(){  
  9.         BankAccount bankAccount = null;  
  10.         try {  
  11.             bankAccount = (BankAccount)super.clone();  
  12.         } catch (CloneNotSupportedException e) {  
  13.             e.printStackTrace();  
  14.         }  
  15.         return bankAccount;  
  16.     }  
  17.       
  18.     public void add(double give){  
  19.         this.balance += give;  
  20.     }  
  21.       
  22.     public void redu(double give){  
  23.         this.balance -= give;  
  24.     }  
  25.       
  26.     public double getBalance() {  
  27.         return balance;  
  28.     }  
  29.     public void setBalance(double balance) {  
  30.         this.balance = balance;  
  31.     }  
  32. }  

package clone; public class BankAccount implements Cloneable{ private double balance; public BankAccount(double ini){ this.balance = ini; } protected Object clone(){ BankAccount bankAccount = null; try { bankAccount = (BankAccount)super.clone(); } catch (CloneNotSupportedException e) { e.printStackTrace(); } return bankAccount; } public void add(double give){ this.balance += give; } public void redu(double give){ this.balance -= give; } public double getBalance() { return balance; } public void setBalance(double balance) { this.balance = balance; } }
        程序入口 TestClone.java 如下:

view plain copy to clipboard print ?
  1. package clone;  
  2. public class TestClone {  
  3.     public static void main(String[] args) {  
  4.         BankAccount bc1 = new BankAccount(1000.0);  
  5.         BankAccount bc2 = (BankAccount) bc1.clone();  
  6.         bc1.add(25);  
  7.         bc2.add(20);  
  8.         System.out.println("用户 1 的帐户余额为:" + bc1.getBalance());  
  9.         System.out.println("用户 2 的帐户余额为:" + bc2.getBalance());  
  10.     }  
  11. }  

package clone; public class TestClone { public static void main(String[] args) { BankAccount bc1 = new BankAccount(1000.0); BankAccount bc2 = (BankAccount) bc1.clone(); bc1.add(25); bc2.add(20); System.out.println("用户 1 的帐户余额为:" + bc1.getBalance()); System.out.println("用户 2 的帐户余额为:" + bc2.getBalance()); } }
        执行 TestClone.java,打印结果如下:
用户 1 的帐户余额为:1025.0
用户 2 的帐户余额为:1020.0

        这下用户 2 把钱存进了自己的帐户,而 Java 程序员也不用头疼了。皆大欢喜。

 

你可能感兴趣的:(关于 Java 中对象的复制)