先看一个简单案例

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. C#.NET Winform 通用开发框架

    C/S系统开发框架-企业版 V4.0 (Enterprise Edition) 简介: http://www.csframework.com/cs-framework-4.0.htm 视频下载: 百度 ...

  2. 写个自己的Xcode4插件

    推荐:http://onevcat.com/2013/02/xcode-plugin/   刚写iOS程序的时候就知道Xcode支持第三方插件,比如ColorSense等很实用的插件,但Xcode的插 ...

  3. Java 内部类种类及使用解析

    package com.learnjava.innerclass; class MemberInner { private int d = 1; private int a = 2; // 定义一个成 ...

  4. 【Java】Java XML 技术专题

    XML 基础教程 XML 和 Java 技术 Java XML文档模型 JAXP(Java API for XML Parsing) StAX(Streaming API for XML) XJ(XM ...

  5. Artem and Array

    Codeforces Round #253 (Div. 1) C:http://codeforces.com/problemset/problem/442/C 题意:给你一个序列,然后你每次可以删除一 ...

  6. Android Wear开发 - 数据通讯 - 第四节 : 数据封装(解决不能序列化问题)

    一. 前言 背景 一开始笔者在研究数据发送与接收的时候,看到Wear数据类DataMap除了可以put基本类型外,还有个fromBundle方法来构建一个DataMap对象.所以一口气的将原本功能上的 ...

  7. Oracle 多版本控制

    SESSION 1: SQL> create table t 2 as 3 select * from all_users; Table created. SQL> variable x ...

  8. 奔跑的xiaodao

    http://acm.hrbust.edu.cn/index.php?m=ProblemSet&a=showProblem&problem_id=2086 很明显的一个二分题目.因为要 ...

  9. lightoj 1291 无向图边双联通+缩点统计叶节点

    题目链接:http://lightoj.com/volume_showproblem.php?problem=1291 #include<cstdio> #include<cstri ...

  10. 关于kali安装vmware的坑,linux套路太深。

    http://www.linuxidc.com/Linux/2015-08/122240.htm 但是还有些坑 安装gcc5.4.1 apt-get install gcc-5 gcc-5所在目录 / ...