詳解java中的深拷貝和淺拷貝(clone()方法的重寫、使用序列化實(shí)現(xiàn)真正的深拷貝)
1.序列化實(shí)現(xiàn)
public class CloneUtils {
@SuppressWarnings("unchecked")
public static <T extends Serializable> T clone(T object){
T cloneObj = null;
try {
ByteArrayOutputStream out = new ByteArrayOutputStream();
ObjectOutputStream obs = new ObjectOutputStream(out);
obs.writeObject(object);
obs.close();
ByteArrayInputStream ios = new ByteArrayInputStream(out.toByteArray());
ObjectInputStream ois = new ObjectInputStream(ios);
cloneObj = (T) ois.readObject();
}catch (Exception e){
e.printStackTrace();
}
return cloneObj;
}
}
2.主代碼
public class TestString {
public static void main(String[] args) {
TestString test = new TestString();
System.out.println("-------淺拷貝---------");
test.qianCopyTest();
System.out.println();
System.out.println("--------使用clone深拷貝--------");
test.defaultCloneTest();
System.out.println();
System.out.println("--------使用序列化實(shí)現(xiàn)對(duì)象的拷貝--------");
test.streamClonrTest();
System.out.println("--------耗時(shí)對(duì)比--------");
System.out.println("耗時(shí)1 : "+ test.qianCopyCost());
System.out.println("耗時(shí)2 : "+ test.CloneCopyCost());
System.out.println("耗時(shí)3 : "+ test.StreamCopyCost());
}
/*淺拷貝*/
private void qianCopyTest() {
String s = "cd";
change(s);
System.out.println(s);
System.out.println("----------------");
String b = new String("cd");
change(b);
System.out.println(b);
System.out.println("----------------");
int me = 1;
change(me);
System.out.println(me);
System.out.println("----------------");
Person person = new Person("我", 13,new Email("我"));
change(person);
System.out.println(person.toString());
}
/*使用默認(rèn)的clone方法,需要Person實(shí)現(xiàn)Cloneable接口*/
private void defaultCloneTest(){
Person person = new Person("我", 13,new Email("我"));
Person person1 = person.clone();
Person person2 = person.clone();
System.out.println("person : 【"+person+"】");
System.out.println("person1 : 【"+person1+"】");
System.out.println("person2 : 【"+person2+"】");
//改一個(gè)就會(huì)觸動(dòng)全部??! 這就是使用默認(rèn)的clone方法的弊端
/*該clone()方法是使用Object類的clone()方法,但是該方法存在一個(gè)缺陷,它并不會(huì)將對(duì)象的所有屬性全部拷貝過來,而是有選擇性的拷貝,基本規(guī)則如下:
1、 基本類型
如果變量是基本很類型,則拷貝其值,比如int、float等。
2、 對(duì)象
如果變量是一個(gè)實(shí)例對(duì)象,則拷貝其地址引用,也就是說此時(shí)新對(duì)象與原來對(duì)象是公用該實(shí)例變量。
3、 String字符串
若變量為String字符串,則拷貝其地址引用。但是在修改時(shí),它會(huì)從字符串池中重新生成一個(gè)新的字符串,原有紫都城對(duì)象保持不變。*/
person.getEmail().setContent("你");
System.out.println("之后的person : 【"+person+"】");
System.out.println("之后的person1 : 【"+person1+"】");
System.out.println("之后的person2 : 【"+person2+"】");
}
/*使用序列化實(shí)現(xiàn)對(duì)象的拷貝,需要對(duì)象以及對(duì)象中的其他對(duì)象都要實(shí)現(xiàn)Serializable接口*/
private void streamClonrTest(){
Person person = new Person("我", 13,new Email("我"));
Person person1 = CloneUtils.clone(person);
Person person2 = CloneUtils.clone(person);
System.out.println("person : 【"+person+"】");
System.out.println("person1 : 【"+person1+"】");
System.out.println("person2 : 【"+person2+"】");
person.getEmail().setContent("你");
System.out.println("之后的person : 【"+person+"】");
System.out.println("之后的person1 : 【"+person1+"】");
System.out.println("之后的person2 : 【"+person2+"】");
}
private static void change(String x) {
x = "ab";
}
private static void change(int x) {
x = 2;
}
private static void change(Person x) {
x = new Person("你", 20, new Email("你"));
}
private long qianCopyCost(){
long start = System.currentTimeMillis();
Person person = new Person("我", 13,new Email("我"));
List<Person> list = new ArrayList<>();
for(int i = 0;i<=10000;i++){
list.add(new Person("你", 20, new Email("你")));
}
return System.currentTimeMillis()-start;
}
private long CloneCopyCost(){
long start = System.currentTimeMillis();
Person person = new Person("我", 13,new Email("我"));
List<Person> list = new ArrayList<>();
for(int i = 0;i<=10000;i++){
list.add(person.clone());
}
return System.currentTimeMillis()-start;
}
private long StreamCopyCost(){
long start = System.currentTimeMillis();
Person person = new Person("我", 13,new Email("我"));
List<Person> list = new ArrayList<>();
for(int i = 0;i<=10000;i++){
list.add(CloneUtils.clone(person));
}
return System.currentTimeMillis()-start;
}
}
class Person implements Serializable, Cloneable {
private static final long serialVersionUID = -8584225043397465132L;
private String name;
private int age;
public void setEmail(Email email) {
this.email = email;
}
private Email email;
public Email getEmail() {
return email;
}
public void setName(String name) {
this.name = name;
}
public void setAge(int age) {
this.age = age;
}
public Person(String name, int age, Email email) {
this.name = name;
this.age = age;
this.email = email;
}
@Override
public String toString() {
return "name : " + name + " | age : " + age +" | content : "+email.getContent();
}
@Override
protected Person clone() {
Person person = null;
try {
person = (Person) super.clone();
/*如果加上下一行 “使用clone深拷貝” 就不會(huì)改一處其他都改變了*/
person.setEmail(new Email(person.getEmail().getContent()));
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
return person;
}
}
class Email implements Serializable {
private static final long serialVersionUID = 1426052929769365539L;
private String content;
public void setContent(String content) {
this.content = content;
}
public String getContent() {
return content;
}
public Email(String content) {
this.content = content;
}
}
測(cè)試了一下時(shí)間:
輸出:
-------淺拷貝---------
cd
cd
1
name : 我 | age : 13 | content : 我
--------使用clone深拷貝--------
person : 【name : 我 | age : 13 | content : 我】
person1 : 【name : 我 | age : 13 | content : 我】
person2 : 【name : 我 | age : 13 | content : 我】
之后的person : 【name : 我 | age : 13 | content : 你】
之后的person1 : 【name : 我 | age : 13 | content : 我】
之后的person2 : 【name : 我 | age : 13 | content : 我】
--------使用序列化實(shí)現(xiàn)對(duì)象的拷貝--------
person : 【name : 我 | age : 13 | content : 我】
person1 : 【name : 我 | age : 13 | content : 我】
person2 : 【name : 我 | age : 13 | content : 我】
之后的person : 【name : 我 | age : 13 | content : 你】
之后的person1 : 【name : 我 | age : 13 | content : 我】
之后的person2 : 【name : 我 | age : 13 | content : 我】
--------耗時(shí)對(duì)比--------
耗時(shí)1 : 2
耗時(shí)2 : 1
耗時(shí)3 : 338
以上所述是小編給大家介紹的java中的深拷貝和淺拷貝(clone()方法的重寫、使用序列化實(shí)現(xiàn)真正的深拷貝)詳解整合,希望對(duì)大家有所幫助,如果大家有任何疑問請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
相關(guān)文章
SpringBoot如何使用@Value取配置文件中的map配置
這篇文章主要介紹了SpringBoot如何使用@Value取配置文件中的map配置方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-05-05
如何用java計(jì)算兩個(gè)時(shí)間相差多少小時(shí)
最近工作中遇到需要計(jì)算時(shí)間差,下面這篇文章主要給大家介紹了關(guān)于如何用java計(jì)算兩個(gè)時(shí)間相差多少小時(shí)的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-12-12
Springboot整合fastdfs實(shí)現(xiàn)分布式文件存儲(chǔ)
本文主要介紹了Springboot整合fastdfs實(shí)現(xiàn)分布式文件存儲(chǔ),詳細(xì)闡述了Springboot應(yīng)用程序如何與FastDFS進(jìn)行集成及演示了如何使用Springboot和FastDFS實(shí)現(xiàn)分布式文件存儲(chǔ),感興趣的可以了解一下2023-08-08
Maven導(dǎo)入Junit4后在test中無法引用問題
在使用Maven進(jìn)行項(xiàng)目管理時(shí),可能會(huì)遇到導(dǎo)入依賴和打開項(xiàng)目結(jié)構(gòu)的問題,本文通過實(shí)際經(jīng)驗(yàn),提供了一些解決方法和技巧,希望能幫助遇到相同問題的開發(fā)者,此外,還鼓勵(lì)大家多多支持和分享個(gè)人經(jīng)驗(yàn),以便于共同進(jìn)步2024-10-10

