浅析深拷贝与浅拷贝以及Object.clone()

深拷贝和浅拷贝的基本概念的理解:

浅拷贝是指拷贝对象时仅仅拷贝对象本身(包括对象中的基本变量),而不拷贝对象包含的引用指向的对象,被复制对象的所有变量都含有与原来的对象相同的值,而所有的对其他对象的引用仍然指向原来的对象

深拷贝不仅拷贝对象本身,而且拷贝对象包含的引用指向的所有对象 

Object.clone()实现

本身实现为浅拷贝

package com.suirui.common.util;

import com.suirui.common.constants.SysConstants;
import org.apache.commons.lang3.StringUtils;

public class TokenUtilTest {
	
	public static void main(String[] args) {

		Cat cat = new Cat("小花");
		Person  p1 = new Person("张三",cat);
		try {
			Person p2 =(Person) p1.clone();
			p2.setName("李四");
			Cat cat1 = p2.getCat();
			cat1.setName("小白");
			p2.setCat(cat1);
			System.out.println(p1);
			System.out.println(p2);
		} catch (CloneNotSupportedException e) {
			e.printStackTrace();
		}
	}



}
class Person implements Cloneable {
	String name;
	Cat cat;

	public Cat getCat() {
		return cat;
	}

	public void setCat(Cat cat) {
		this.cat = cat;
	}

	public Person(String name,Cat cat) {
		this.name = name;
		this.cat = cat;
	}

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}


	@Override
	protected Object clone() throws CloneNotSupportedException {
		return super.clone();
	}



	@Override
	public String toString() {
		return "Person{" +
				"name='" + name + '\'' +
				", cat=" + cat +
				'}';
	}
}

class Cat{
	String name;

	public Cat(String name) {
		this.name = name;
	}

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

	@Override
	public String toString() {
		return "Cat{" +
				"name='" + name + '\'' +
				'}';
	}
}

从结果可以看出来,clone()是浅copy。

实现深拷贝一

package com.suirui.common.util;

import com.suirui.common.constants.SysConstants;
import org.apache.commons.lang3.StringUtils;

public class TokenUtilTest {
	
	
	public static void main(String[] args) {

		Cat cat = new Cat("小花");
		Person  p1 = new Person("张三",cat);
		try {
			Person p2 =(Person) p1.clone();
			p2.setName("李四");
			Cat cat1 = p2.getCat();
			cat1.setName("小白");
			p2.setCat(cat1);
			System.out.println(p1);
			System.out.println(p2);
		} catch (CloneNotSupportedException e) {
			e.printStackTrace();
		}
	}



}
class Person implements Cloneable {
	String name;
	Cat cat;

	public Cat getCat() {
		return cat;
	}

	public void setCat(Cat cat) {
		this.cat = cat;
	}

	public Person(String name,Cat cat) {
		this.name = name;
		this.cat = cat;
	}

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}


	@Override
	protected Object clone() throws CloneNotSupportedException {
		Person p = (Person) super.clone();
		p.cat = (Cat) this.cat.clone();
		return p;
	}



	@Override
	public String toString() {
		return "Person{" +
				"name='" + name + '\'' +
				", cat=" + cat +
				'}';
	}
}

class Cat implements Cloneable{
	String name;

	public Cat(String name) {
		this.name = name;
	}

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

	@Override
	protected Object clone() throws CloneNotSupportedException {
		return super.clone();
	}

	@Override
	public String toString() {
		return "Cat{" +
				"name='" + name + '\'' +
				'}';
	}
}

这种方式比较繁琐,要将所有引用类都进行克隆一次。

实现深拷贝二

在Java语言里深复制一个对象,常常可以先使对象实现Serializable接口,然后把对象(实际上只是对象的一个拷贝)写到一个流里,再从流里读出来,便可以重建对象。

注意:

  • 写在流里的是对象的一个拷贝,而原对象仍然存在于JVM里面
  • 对象以及对象内部所有引用到的对象都是可序列化的
  • 如果不想序列化,则需要使用transient来修饰
package com.suirui.common.util;

import com.suirui.common.constants.SysConstants;
import org.apache.commons.lang3.StringUtils;

import java.io.*;

public class TokenUtilTest {
  
    public static void main(String[] args) {

        Cat cat = new Cat("小花");
        Person p1 = new Person("张三", cat);
        try {
            Person p2 = (Person) p1.clone();
            p2.setName("李四");
            Cat cat1 = p2.getCat();
            cat1.setName("小白");
            p2.setCat(cat1);
            System.out.println(p1);
            System.out.println(p2);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }


}

class Person implements Serializable {
    String name;
    Cat cat;

    public Cat getCat() {
        return cat;
    }

    public void setCat(Cat cat) {
        this.cat = cat;
    }

    public Person(String name, Cat cat) {
        this.name = name;
        this.cat = cat;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }


    @Override
    protected Object clone() {

        try {
            //序列化对象
            ByteArrayOutputStream bos = new ByteArrayOutputStream();
            ObjectOutputStream oos = new ObjectOutputStream(bos);
            oos.writeObject(this);
            //反序列化对象
            ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray());
            ObjectInputStream ois = new ObjectInputStream(bis);
            return ois.readObject();
        } catch (Exception e) {
            e.printStackTrace();
            return null;
        }
    }


    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", cat=" + cat +
                '}';
    }
}

class Cat implements Serializable {
    String name;

    public Cat(String name) {
        this.name = name;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    @Override
    public String toString() {
        return "Cat{" +
                "name='" + name + '\'' +
                '}';
    }
}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值