深拷贝和浅拷贝-留坑

6-4 浅拷贝与深拷贝问题 (10分)

Java中的对象拷贝(Object Copy)指的是将一个对象的所有属性(成员变量)拷贝到另一个有着相同类类型的对象中去。举例说明:比如,对象A和对象B都属于类S,具有属性a和b。那么对对象A进行拷贝操作赋值给对象B就是:B.a=A.a; B.b=A.b;
浅拷贝(Shallow Copy):①对于数据类型是基本数据类型的成员变量,浅拷贝会直接进行值传递,也就是将该属性值复制一份给新的对象。因为是两份不同的数据,所以对其中一个对象的该成员变量值进行修改,不会影响另一个对象拷贝得到的数据。②对于数据类型是引用数据类型的成员变量,比如说成员变量是某个数组、某个类的对象等,那么浅拷贝会进行引用传递,也就是只是将该成员变量的引用值(内存地址)复制一份给新的对象。因为实际上两个对象的该成员变量都指向同一个实例。在这种情况下,在一个对象中修改该成员变量会影响到另一个对象的该成员变量值。为了解决这个问题,引入深拷贝:

深拷贝 对于引用类型成员变量,比如数组或者类对象,深拷贝会在目标对象内新建一个对象空间,然后拷贝原对象对应成员变量实体对象里面的内容,所以它们指向了不同的内存空间。改变其中一个,不会对另外一个也产生影响。
在Java中 有一个方法为protected Object clone() throws CloneNotSupportedException,这个方法就是进行的浅拷贝。某个类要实现深拷贝 这个类需要实现Cloneable接口,然后对clone方法进行重写

重写clone方法定义:

class Car implements Cloneable{
    
      //定义一个Car类 实现接口Cloneable
       //成员变量定义
   public Object clone(){
    
      //重写clone方法
       Car c = null;
        try {
    
    
                c = (Car)super.clone(); //调用父类Object实现浅拷贝
               } catch (CloneNotSupportedException e) {
    
    
                        e.printStackTrace();
               }
        	//编写代码实现深拷贝
		 
      }
    //编写代码实现其他成员方法
}

Car 类中包含
1.属性:

private String name;
private CarDriver driver;
private int[] scores;

2.无参构造函数

public Car() {
    
    
}

3.成员方法:

@Override
public String toString() {
    
    
    return "Car [name=" + name + ", driver=" + driver + ", scores=" + Arrays.toString(scores) + "]";
}

4.其他get和set成员方法:public String getName() ,public CarDriver getDriver(),public int[] getScores()和
public void setName(String name),public void setDriver(CarDriver dr),public void setScores(int[] s)
clone方法实现深拷贝

CarDriver为已定义好的类,如下:

class CarDriver {
    
    
    private String name;
    public String getName() {
    
    
		return name;
	}
	public void setName(String name) {
    
    
		this.name = name;
	}
	public CarDriver() {
    
    }
	public String toString() {
    
    
	    return "CarDriver [name=" + name+"]";
	}
}

Car 类的完整代码 也就是需要进行操作的代码块

class Car implements Cloneable{
    
      //定义一个Car类 实现接口Cloneable
    //成员变量定义
    private String name;
    private CarDriver driver;
    private int[] scores;

    public Car() {
    
    
    }

    public Object clone(){
    
      //重写clone方法
        Car c = null;
        try {
    
    
            c = (Car)super.clone(); //调用父类Object实现浅拷贝
        } catch (CloneNotSupportedException e) {
    
    
            e.printStackTrace();
        }
        //编写代码实现深拷贝
        if(getScores()!=null) {
    
    
        	// 这里是深克隆的
            c.setScores(getScores().clone());
        } else {
    
    
            c.setScores(null);
        }

        if(getDriver()!=null) {
    
    
        	// 这里开辟了新空间
            c.setDriver(new CarDriver());
            c.getDriver().setName(driver.getName());
        }
        return c;
    }
    //编写代码实现其他成员方法
    @Override
    public String toString() {
    
    
        return "Car [name=" + name + ", driver=" + driver + ", scores=" + Arrays.toString(scores) + "]";
    }

    public String getName() {
    
    
        return name;
    }

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

    public CarDriver getDriver() {
    
    
        return driver;
    }

    public void setDriver(CarDriver driver) {
    
    
        this.driver = driver;
    }

    public int[] getScores() {
    
    
        return scores;
    }

    public void setScores(int[] scores) {
    
    
        this.scores = scores;
    }
}

对值类型的属性 浅克隆的时候是开辟了新空间的

对引用对象的浅克隆就指向了同一个地址

所以对于这种对象我们手动开辟空间

像数组 对象 这种 就要手动开辟空间

大概就理解这些好了

多了我也不懂

猜你喜欢

转载自blog.csdn.net/weixin_45543674/article/details/109228339
今日推荐