先看一个简单案例

public class Test {
    public static void main(String args[]) {
        Student stu1 = new Student();
        stu1.number = 1;
        Student stu2 = stu1;//stu1和stu2指向堆内存中同一个对象
        System.out.println("学生1:" + stu1.number + "    学生2:" + stu2.number);
        stu2.number = 2;
        System.out.println("学生1:" + stu1.number + "    学生2:" + stu2.number);
        System.out.println(stu1 == stu2);
    }
}
class Student {
    public int number;
}

如上这种形式,仅仅将一个对象的引用赋给另一个引用,并没有复制出另一个对象,这两个引用指向的是内存中的同一个对象
所以,这根本不叫对象的复制

对象的复制(浅复制)

public class Test {
    public static void main(String args[]) {
        Student stu1 = new Student();
        stu1.number = 1;
        Student stu2 = (Student) stu1.clone();
        System.out.println("学生1:" + stu1.number + "    学生2:" + stu2.number);
        stu2.number = 2;
        System.out.println("学生1:" + stu1.number + "    学生2:" + stu2.number);
        System.out.println(stu1 == stu2);
    }
}
class Student implements Cloneable {
    public int number;
    @Override
    public Object clone() {
        Student stu = null;
        try {
            stu = (Student) super.clone();
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        return stu;
    }
}

Object有个protected的clone方法,该方法的签名是:
protected native Object clone() throws CloneNotSupportedException;
要想对一个对象进行复制,就需要覆盖clone方法,一般步骤是(浅复制):
  • 1、被复制的类需要实现Clonenable接口,不实现的话在调用clone方法会抛出CloneNotSupportedException异常,该接口为标记接口(不含任何方法)
  • 2、覆盖clone()方法,访问修饰符设为public。方法中调用super.clone()方法得到需要的复制对象(native为本地方法)

浅复制可能出现的问题

public class Test {
    public static void main(String args[]) {
        Student stu1 = new Student();
        stu1.number = 1;
        stu1.address = new Address("广州");
        stu1.courseList.add("语文");

        //复制stu1
        Student stu2 = (Student) stu1.clone();
        System.out.println("学生1:" + stu1.number + "    " + stu1.address + "    " + stu1.courseList);
        System.out.println("学生2:" + stu2.number + "    " + stu2.address + "    " + stu2.courseList);

        //更改stu2
        stu2.number = 2;
        stu2.address = new Address("深圳");
        stu2.courseList.add("数学");
        System.out.println("学生1:" + stu1.number + "    " + stu1.address + "    " + stu1.courseList);
        System.out.println("学生2:" + stu2.number + "    " + stu2.address + "    " + stu2.courseList);
    }
}
class Student implements Cloneable {
    public int number;
    public ArrayList<String> courseList = new ArrayList<String>();
    public Address address;
    @Override
    public Object clone() {
        Student stu = null;
        try {
            stu = (Student) super.clone();
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        return stu;
    }
}
class Address {
    public String add;
    public Address(String add) {
        this.add = add;
    }
}

更改stu2的address和courseList后,是不是发现,stu1的相应值也变了?这是为什么呢?
这是因为Java做了一个偷懒的拷贝动作,Object类提供的方法clone只是拷贝本对象,其对象内部的数组、引用对象等都不拷贝,还是指向原生对象的内部元素的地址,这种拷贝就叫做浅拷贝。
确实是非常浅,两个对象共享了一个私有变量,你改我改大家都能改,是一个种非常不安全的方式,在实际项目中使用还是比较少的。
注意:八大基本类型和String(虽然String 是引用类型,但Java其实是希望你把它也"认为"是基本类型,String 是没有 clone 方法的)等都会被拷贝的。

深复制

public class Test {
    public static void main(String args[]) {
        Student stu1 = new Student();
        stu1.number = 1;
        stu1.address = new Address("广州");
        stu1.courseList.add("语文");
        //复制stu1
        Student stu2 = (Student) stu1.clone();
        System.out.println("学生1:" + stu1.number + "    " + stu1.address + "    " + stu1.courseList);
        System.out.println("学生2:" + stu2.number + "    " + stu2.address + "    " + stu2.courseList);
        //更改stu2
        stu2.number = 2;
        stu2.address = new Address("深圳");
        stu2.courseList.add("数学");
        System.out.println("学生1:" + stu1.number + "    " + stu1.address + "    " + stu1.courseList);
        System.out.println("学生2:" + stu2.number + "    " + stu2.address + "    " + stu2.courseList);
    }
}
class Student implements Cloneable {
    public int number;
    public ArrayList<String> courseList = new ArrayList<String>();
    public Address address;
    @SuppressWarnings("unchecked")
    @Override
    public Object clone() {
        Student stu = null;
        try {
            stu = (Student) super.clone();//浅复制 
            stu.address = (Address) address.clone(); //深度复制
            stu.courseList = (ArrayList<String>) courseList.clone();//深度复制
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        return stu;
    }
}
class Address implements Cloneable {//为了达到真正的复制对象,而不是纯粹引用复制。我们需要将Address类可复制化
    public String add;
    public Address(String add) {
        this.add = add;
    }
    @Override
    public Object clone() {
        Address addr = null;
        try {
            addr = (Address) super.clone();
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        return addr;
    }
}


Java中的浅复制和深复制 Cloneable clone的更多相关文章

  1. JAVA中浅复制与深复制 - coolmist - ITeye技术网站

    body{ font-family: "Microsoft YaHei UI","Microsoft YaHei",SimSun,"Segoe UI& ...

  2. Java中引用的浅复制和深复制

    Java中除了基本类型int,char,double等的赋值是按照值传递之外,其余的类型和对象都是按照引用进行传递的. 下面来看一个关于引用的例子. package referenceCopy;// ...

  3. JAVA中浅复制与深复制

    1.浅复制与深复制概念⑴浅复制(浅克隆)被复制对象的所有变量都含有与原来的对象相同的值,而所有的对其他对象的引用仍然指向原来的对象.换言之,浅复制仅仅复制所考虑的对象,而不复制它所引用的对象. ⑵深复 ...

  4. Java中的“浅复制”与“深复制”

    复制 将一个对象的引用复制给另一个对象,一共有三种方式.第一种方式是直接赋值,第二种方式是浅复制,第三种方式是深复制. 1.直接赋值 在Java中,A a1 = a2,这实际上复制的是引用,也就是说 ...

  5. java基础-浅复制与深复制的理解

    浅复制与深复制在很多编程语言中都有出现,那么什么是浅复制,什么是深复制呢? 要区分浅复制与深复制,首先我们要明确什么是复制,怎样才算是复制.复制的例子在生活中也随处可见,如复印一份文档,复制一段文字等 ...

  6. Java里的浅复制与深复制

    1.浅复制与深复制概念 ⑴浅复制(浅克隆) 被复制对象的所有变量都含有与原来的对象相同的值,而所有的对其他对象的引用仍然指向原来的对象.换言之,浅复制仅仅复制所考虑的对象,而不 复制它所引用的对象. ...

  7. 深度解析javascript中的浅复制和深复制

    原文:深度解析javascript中的浅复制和深复制 在谈javascript的浅复制和深复制之前,我们有必要在来讨论下js的数据类型.我们都知道有Number,Boolean,String,Null ...

  8. js中的浅复制和深复制

    浅复制:浅复制是复制引用,复制后的引用都是指向同一个对象的实例,彼此之间的操作会互相影响 深复制:深复制不是简单的复制引用,而是在堆中重新分配内存,并且把源对象实例的所有属性都进行新建复制,以保证深复 ...

  9. Python中的浅复制、深复制

    参考 https://docs.python.org/3/library/copy.html?highlight=copy%20copy#copy.copy https://en.wikipedia. ...

  10. php对象复制、clone、浅复制与深复制实例详解

    php对象复制.clone.浅复制与深复制实例详解 一.用clone(克隆)来复制对象$obj1 = new Object();$obj2 = clone $obj1;clone方法会触发对象里定义的 ...

随机推荐

  1. 那些年被我坑过的Python——摩拳擦掌(第三章)

    集合类型: 集合类型中的元素是唯一的! 集合的定义与赋值: set_1 = set([1, 3, 5, 7, 2]) set_2 = set([2, 4, 6, 8, 3]) 集合的运算操作 # 交集 ...

  2. Redux1

    Redux 写在前面 写React也有段时间了,一直也是用Redux管理数据流,最近正好有时间分析下源码,一方面希望对Redux有一些理论上的认识:另一方面也学习下框架编程的思维方式. Redux如何 ...

  3. Ruby自学笔记(五)— 条件判断

    条件判断,在编程语言中都存在,而Ruby中的条件判断和Java中类似,当然还是存在些许差别 Ruby中条件判断的条件: 1) 可以使用 ==,<,>等比较运算来作为条件,比较运算可以返回t ...

  4. HTML/W3C-WHATWG-Differences

    Differences between the W3C HTML 5.1 specification and the WHATWG LS The W3C HTML 5.1 specification: ...

  5. 【Java】Java网络编程菜鸟进阶:TCP和套接字入门

    Java网络编程菜鸟进阶:TCP和套接字入门 JDK 提供了对 TCP(Transmission Control Protocol,传输控制协议)和 UDP(User Datagram Protoco ...

  6. c# 循环语句练习题;

    1. 求100以内质数的和 2. 兔子问题 3. 九九乘法表:   一行一行打印: 4. 有一张超大的纸:   纸质的厚度是0.01:   对折多少次,可以达到珠峰的高度:   按照8848来计算: ...

  7. 【HDOJ】2217 Visit

    挺好的一道DP. /* 2217 */ #include <iostream> #include <cstdio> #include <cstring> #incl ...

  8. BZOJ 1030 [JSOI2007]文本生成器

    1030: [JSOI2007]文本生成器 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 2624  Solved: 1087[Submit][Stat ...

  9. 「Poetize7」Freda的访客

    描述 Description 小猫们看到蛋糕比饼干大之后,普遍认为蛋糕比饼干要好>.<.所以,如果Freda 给了第i 只小猫蛋糕且这个小猫是第一个吃到蛋糕的,那么就必须给第i+2,i+4 ...

  10. HDU 3829 Cat VS Dog(最大独立集)

    题目大意: 有n只猫,有m只狗.现在有P个学生去参观动物园.每个孩子有喜欢的动物和不喜欢的动物.假如他喜欢猫那么他就一定不喜欢狗(反之亦然). 如果一个孩子喜欢一个动物,那么这个动物不会被移除,若是不 ...