java深克隆与浅克隆的区别原理及其实现_TO UP!的博客-程序员秘密

技术标签: java基础  java  

一、为什么要克隆?

答案是:克隆的对象可能包含一些已经修改过的属性,保留着你想克隆对象的值,而new出来的对象的属性全是一个新的对象,对应的属性没有值,所以我们还要重新给这个对象赋值。即当需要一个新的对象来保存当前对象的“状态”就靠clone方法了。那么我把这个对象的临时属性一个一个的赋值给我新new的对象不也行嘛?可以是可以,但是一来麻烦不说,二来,大家通过上面的源码都发现了clone是一个native方法,就是快啊,在底层实现的。
 

二、如何实现克隆

分三步:

  1. 对象的类实现Cloneable接口;
  2. 覆盖Object类的clone()方法 (覆盖clone()方法,访问修饰符设为public,默认是protected);
  3. 在clone()方法中调用super.clone();

三、两种不同的克隆方法,浅克隆(ShallowClone)和深克隆(DeepClone)。

浅克隆是指拷贝对象时仅仅拷贝对象本身(包括对象中的基本变量),而不拷贝对象包含的引用指向的对象。

深克隆不仅拷贝对象本身,而且拷贝对象包含的引用指向的所有对象。举例来说更加清楚:

浅克隆

package cn.zhm.day4;
 
public class Student implements Cloneable {
	private int age;
	private String name;
 
	public Student(int age, String name) {
		this.age = age;
		this.name = name;
	}
 
	public int getAge() {
		return age;
	}
 
	public void setAge(int age) {
		this.age = age;
	}
 
	public String getName() {
		return name;
	}
 
	public void setName(String name) {
		this.name = name;
	}
 
	@Override
	public String toString() {
		return "Student [age=" + age + ", name=" + name + "]";
	}
 
	@Override
	public Object clone() throws CloneNotSupportedException {
		// TODO Auto-generated method stub
		return super.clone();
	}
 
	/**
	 * @param args
	 * @throws CloneNotSupportedException
	 */
	public static void main(String[] args) throws CloneNotSupportedException {
		Student student1 = new Student(20, "张三");
		Student student2 = (Student) student1.clone();
		student2.setAge(22);// 注意修改student2的age值 但是没有影响 student1的值
		System.out.println("student1:" + student1.getName() + "-->"+ student1.getAge());
		System.out.println("student2:" + student2.getName() + "-->"+ student2.getAge());
 
	}
}
运行结果:
student1:张三-->20
student2:张三-->22 

引入深克隆,浅入克隆导致的问题

package cn.zhm.day4;
 
class Teacher implements Cloneable {
	private String name;
	private Student student;
	
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	
	public Student getStudent() {
		return student;
	}
	public void setStudent(Student student) {
		this.student = student;
	}
	@Override
	public String toString() {
		return "Teacher [name=" + name + ", student=" + student + "]";
	}
	
	@Override
	public Object clone() throws CloneNotSupportedException {
		// TODO Auto-generated method stub
		return super.clone();
	}
	public static void main(String[] args) throws CloneNotSupportedException {
		Student s1 = new Student();
		s1.setAge(20);
		s1.setName("张三");
		Teacher teacher1 = new Teacher();
		teacher1.setName("小赵老师");
		teacher1.setStudent(s1);
		//为什么会出现以下结果, Teacher中的clone方法
		Teacher teacher2 = (Teacher)teacher1.clone();
		Student s2 = teacher2.getStudent();
		s2.setName("李四");
		s2.setAge(30);
		System.out.println("teacher1:"+teacher1);
		System.out.println("teacher2:"+teacher2);
		
	}
	
}

运行结果:

teacher1:Teacher [name=小赵老师, student=Student [age=30, name=李四]]
teacher2:Teacher [name=小赵老师, student=Student [age=30, name=李四]

 

深克隆

要克隆的类和类中所有非基本数据类型的属性对应的类

必需在clone()方法注意点

package cn.zhm.day4;
 
class Teacher implements Cloneable {
    private String name;
    private Student student;
    
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    
    public Student getStudent() {
        return student;
    }
    public void setStudent(Student student) {
        this.student = student;
    }
    @Override
    public String toString() {
        return "Teacher [name=" + name + ", student=" + student + "]";
    }
    
    @Override
    public Object clone() throws CloneNotSupportedException {
        // TODO Auto-generated method stub
        //注意以下代码
        Teacher teacher = (Teacher)super.clone();
        teacher.setStudent((Student)teacher.getStudent().clone());
        return teacher;
    }
    public static void main(String[] args) throws CloneNotSupportedException {
        Student s1 = new Student();
        s1.setAge(20);
        s1.setName("张三");
        Teacher teacher1 = new Teacher();
        teacher1.setName("小赵老师");
        teacher1.setStudent(s1);
        Teacher teacher2 = (Teacher)teacher1.clone();
        teacher2.setName("小明老师");
        Student s2 = teacher2.getStudent();
        s2.setName("李四");
        s2.setAge(30);
        System.out.println("teacher1:"+teacher1);
        System.out.println("teacher2:"+teacher2);
        
    }
    
}

运行结果:

teacher1:Teacher [name=小赵老师, student=Student [age=20, name=张三]]

teacher2:Teacher [name=小明老师, student=Student [age=30, name=李四]]

好好理解上面代码一定可以掌握 浅克隆(ShallowClone)和深克隆(DeepClone)
 

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/qq_41967563/article/details/104974789

智能推荐

Socket之UDP通信_moluchase的博客-程序员秘密

简单记之,留后用先实现一对一通信服务器类public class UDPServer { public static void main(String[] args) throws IOException { /* * 服务器接收数据 */ //创建DatagramSockte对象,并指定端口号 DatagramSocket socket=new Da

Spring Cloud Alibaba 2.2 服务消费者(Feign)_服务消费者日志 规范_Flying9001的博客-程序员秘密

文章目录1 摘要2 核心 Maven 依赖3 核心代码3.1 application 配置文件3.2 Service 层-服务调用3.3 Controller 控制层3.4 SpringBoot 启动类3.5 其他相关代码4 请求测试4.1 GET 方式请求接口4.2 POST 方式请求接口5 推荐参考资料6 本次提交记录​​1 摘要Feign 是微服务之间相互调用使用最广泛的工具,其内部集成了 Ribbon ,直接在接口上使用注解,极大地简化了微服务的调用过程。本文将介绍基于 Spring Clo

git将某分支的某次提交合并到别的分支_webRambler的博客-程序员秘密

目录1.使用命令行2.使用图形化工具Sourcetree今天我们来学习一下git,git在我们进行代码托管时的首选,所以了解并学会使用git是每个开发人员的必备技能。在进行代码开发的时候,有时需要把某分支(比如dev分支)的某一次提交合并到另一分支(比如hotfix分支),这就需要用到git cherry-pick命令。1.使用命令行首先,切换到dev分支,敲 git lo...

U_BOOT_CMD分析与定制_Huskar_Liu的博客-程序员秘密

U_BOOT_CMD(name,maxargs,repeatable,command,“usage”,“help”)各个参数的意义如下:name:命令名,非字符串,但在U_BOOT_CMD中用“#”符号转化为字符串maxargs:命令的最大参数个数repeatable:是否自动重复(按Enter键是否会重复执行)command:该命令对应的响应函数指针usage:简短的使用说明(字符串...

基于PaaS平台开发流程审批框架界面设计方案(草稿)_多条信息多条事项需要审核界面设计_肖永威的博客-程序员秘密

1、概述        基于PaaS平台开发流程审批业务功能,首先要体现云计算PaaS平台的特性,例如多租户、统一平台、技术规范等。因此,为了达到快速、规范化开发业务功能的要求,在此设计流程审批界面框架,对业务数据和流程数据进行解耦。2、流程审批界面需求分析2.1、审批界面样例分析        1、以审批意见为主体关注点的审批界面,例如下图所示公文审批稿签界面,在业务流转过程中,

leetcode10——正则表达式匹配_Jveyvey的博客-程序员秘密

给你一个字符串s和一个字符规律p,请你来实现一个支持 '.'和'*'的正则表达式匹配。'.' 匹配任意单个字符'*' 匹配零个或多个前面的那一个元素所谓匹配,是要涵盖整个字符串s的,而不是部分字符串。示例 1:输入:s = "aa" p = "a"输出:false解释:"a" 无法匹配 "aa" 整个字符串。示例 2:输入:s = "aa" p = "a*"输出:true解释:因为 '*' 代表可以匹配零个或多个前面的那一个元素, 在这里前面的元素就是 ...

随便推点

初学js之多组图片切换实例_a870066721的博客-程序员秘密

需求是以上效果展示。话不多说,直接代码显示,不涉及代码优化。已实现功能为目的。先看html部分:<body> <div class="dream" id="dream"> <div class="top"> <input type="button" value="上一组"&gt...

Javaee介绍_zflovecf的博客-程序员秘密

1、  为什么需要JavaEE我们编写的JSP代码中,由于大量的显示代码和业务逻辑混淆在一起,彼此嵌套,不利于程序的维护和扩展。当业务需求发生变化的时候,对于程序员和美工都是一个很重的负担。为了程序的易维护性和可扩展性,这就需要我们使用JavaEE技术来进行项目开发2、  什么是JavaEEJavaEE是一个开发分布式企业级应用的规范和标准。Java语言的平台有3个版本:适

学生信息管理系统代码_iqingchun的博客-程序员秘密

#!/usr/bin/python#!coding:utf-8import sys#定义函数,系统菜单信息,帮助用户选择def printStuSys(): print('**********学生信息查询系统***********') print(' 1.查询学生信息 ') print(' 2.增加学生信息 ...

设计模式的分类和六大原则_分类模块时的原则六字_Android成长笔记的博客-程序员秘密

设计模式的六大原则1、开闭原则(Open Close Principle)开闭原则就是说对扩展开放,对修改关闭。在程序需要进行拓展的时候,不能去修改原有的代码,实现一个热插拔的效果。所以一句话概括就是:为了使程序的扩展性好,易于维护和升级。想要达到这样的效果,我们需要使用接口和抽象类,后面的具体设计中我们会提到这点。2、里氏代换原则(Liskov Substit

爬虫python怎么下载_在网上下了一个 python 爬虫程序,怎么运行?_weixin_39780962的博客-程序员秘密

第 1 条附言 · 2016-07-02 18:51:15 +08:00我换 linux 环境现在可以运行了,运行$ python doubanSpider.py 后一直在下载,是什么意思额?/usr/local/lib/python2.7/dist-packages/bs4/__init__.py:166: UserWarning: No parser was explicitly spec...

Flink-安装部署及部署模式介绍_flink安装部署_安然烟火的博客-程序员秘密

Flink支持三大部署模式:1. Local 本地部署Flink 可以运行在 Linux、Mac OS X 和 Windows 上。本地模式的安装唯一需要的只是Java 1.7.x或更高版本,本地运行会启动Single JVM,主要用于测试调试代码。2. Standalone Cluster集群部署Flink自带了集群模式Standalone,这个模式对软件有些要求:1.安装Java1.8或者更高版本2.集群各个节点需要ssh免密登录3. Flink ON YARNFlink ON YAR

推荐文章

热门文章

相关标签