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;
}
}
对值类型的属性 浅克隆的时候是开辟了新空间的
对引用对象的浅克隆就指向了同一个地址
所以对于这种对象我们手动开辟空间
像数组 对象 这种 就要手动开辟空间
大概就理解这些好了
多了我也不懂