一看就懂的,java深拷贝浅拷贝

前言

这两天,男票兴奋地通知我,我的博客终于有排名了,刚好是20000名,原来都是千里之外。我也比较兴奋,在这里谢谢每一个看到我文章的同学。O(∩_∩)O哈哈~,为什么有一种颁奖典礼的赶脚。真的啦,虽然我的博客写的都是比较浅显的基础知识,但是也是一字一字马上去的,有时候为了画图辅助说明,也是费很多时间的。O(∩_∩)O哈哈~,我写博客的目的,就是希望每一个看我博客的人,每看一篇文章都能不用费很多时间,把文章里面的东西弄懂。虽不是大牛,但是我会继续努力的。

话不多说,进入今天的主题吧。今天,依然要记录和分享的是一个非常基础而重要的知识,深拷贝,浅拷贝。我已经记不清楚,已经多少次面试时被问到这个问题了。所以,亲们,来吧,不懂的一定要弄懂哇!

将一个对象的引用复制给另外一个对象,一共有三种方式。第一种方式是直接赋值,第二种方式是浅拷贝,第三种是深拷贝。所以大家知道了哈,这三种概念实际上都是为了拷贝对象啊。

1、直接赋值

好,下面我们先看第一种方式,直接赋值。在Java中,A a1 = a2,我们需要理解的是这实际上复制的是引用,也就是说a1和a2指向的是同一个对象。因此,当a1变化的时候,a2里面的成员变量也会跟着变化。各位,请看下面的代码吧!

package interfaces.nesting;

/* 建立类 */
class Resume {
	private String name;  //姓名
	private String sex;   //性别
	private int age;      //年龄
	private String experience; //工作经历

	public Resume(String name, String sex, int age) {
		this.name = name;
		this.sex = sex;
		this.age = age;
	}

	public void setAge(int age) {
		this.age = age;
	}
	public int getAge() {
		return age;
	}

	public void setExperience(String experience) {
		this.experience = experience;
	}
	public String getExperience() {
		return experience;
	}

	public void displayResume() {
		System.out.println("姓名:"+name+" 性别:"+sex+" 年龄:"+age);
		System.out.println("工作经历:"+experience);
	}
}

public class MainClass {
	public static void main(String[] args) {
		Resume zhangsan = new Resume("zhangsan","男",24);
		zhangsan.setExperience("2009-2013就读于家里蹲大学,精通JAVA,C,C++,C#等代码复制");
		zhangsan.displayResume();
		Resume zhangsan1 = zhangsan;
		zhangsan1.setExperience("2009-2013就读于家里蹲大学,精通JAVA,C,C++,C#等");
		zhangsan.displayResume();
		zhangsan1.displayResume();
	}
}

程序运行结果

姓名:zhangsan 性别:男 年龄:24
工作经历:2009-2013就读于家里蹲大学,精通JAVA,C,C++,C#等代码复制
姓名:zhangsan 性别:男 年龄:24
工作经历:2009-2013就读于家里蹲大学,精通JAVA,C,C++,C#等
姓名:zhangsan 性别:男 年龄:24
工作经历:2009-2013就读于家里蹲大学,精通JAVA,C,C++,C#等

在本程序中,生成了一份zhangsan的简历。之后又复制了一份简历zhangsan1,可见zhangsan1中工作经历发生变化时,zhangsan的工作经历也发生了变化。

2、浅拷贝

上面直接赋值的结果,有时候可能并不是我们所想要的。就像我们投简历的时候,可能会根据应聘公司的类型做出相应的调整,如果是投技术类的工作可能会偏技术一点;如果是投国企啊什么之类的,社会经历学生工作什么的可能也是很重要的一部分。所以我们不需要当我们修改一份简历的时候,所有的简历都变调。不然到时候投技术类的公司又得改回来。说了这么多,我们也就是希望,把a1赋值给a2之后,a1和a2能保持独立,不要互相影响。

实现上面想法之一的方法就是Object的Clone()函数了。在这里,我们需要了解clone()主要做了些什么,创建一个新对象,然后将当前对象的非静态字段复制到该新对象,如果字段是值类型的,那么对该字段执行复制;如果该字段是引用类型的话,则复制引用但不复制引用的对象。因此,原始对象及其副本引用同一个对象。

好,我们先看这一段话的前一部分,如果字段是值类型,则直接复制。如下面程序所示

package interfaces.nesting;

/* 建立类,实现Clone方法  */
class Resume  implements Cloneable{
	private String name;  //姓名
	private String sex;   //性别
	private int age;      //年龄
	private String experience; //工作经历

	public Resume(String name, String sex, int age) {
		this.name = name;
		this.sex = sex;
		this.age = age;
	}

	public void setAge(int age) {
		this.age = age;
	}
	public int getAge() {
		return age;
	}

	public void setExperience(String experience) {
		this.experience = experience;
	}
	public String getExperience() {
		return experience;
	}

	public void displayResume() {
		System.out.println("姓名:"+name+" 性别:"+sex+" 年龄:"+age);
		System.out.println("工作经历:"+experience);
	}

	public Object clone() {
		try {
			return (Resume)super.clone();
		} catch (Exception e) {
			e.printStackTrace();
			return null;
		}
	}
}

public class MainClass {
	public static void main(String[] args) {
		Resume zhangsan = new Resume("zhangsan","男",24);
		zhangsan.setExperience("2009-2013就读于家里蹲大学,精通JAVA,C,C++,C#等代码拷贝和粘贴");
		zhangsan.displayResume();
		Resume zhangsan1 = (Resume)zhangsan.clone();
		zhangsan1.setAge(23);
		zhangsan1.displayResume();
		Resume zhangsan2 = (Resume)zhangsan.clone();
		zhangsan2.setExperience("2009-2013就读于家里蹲大学,精通JAVA,C,C++,C#等代码");
		zhangsan2.displayResume();
		zhangsan.displayResume();
	}
}

程序运行结果

姓名:zhangsan 性别:男 年龄:24
工作经历:2009-2013就读于家里蹲大学,精通JAVA,C,C++,C#等代码拷贝和粘贴
姓名:zhangsan 性别:男 年龄:23
工作经历:2009-2013就读于家里蹲大学,精通JAVA,C,C++,C#等代码拷贝和粘贴
姓名:zhangsan 性别:男 年龄:24
工作经历:2009-2013就读于家里蹲大学,精通JAVA,C,C++,C#等代码
姓名:zhangsan 性别:男 年龄:24
工作经历:2009-2013就读于家里蹲大学,精通JAVA,C,C++,C#等代码拷贝和粘贴

由程序的运行结果可以看出,我们实现了a1和a2引用的独立。

但是什么叫“如果该字段是引用类型的话,则复制引用但不复制引用的对象。因此,原始对象及其副本引用同一个对象。”,到底什么意思?不用着急,我们接下来看下面一段程序:

package interfaces.nesting;

class Experience {

	private String educationBackground;
	private String skills;

	public void setExperience(String educationBackground, String skills) {
		// TODO Auto-generated constructor stub
		this.educationBackground = educationBackground;
		this.skills = skills;
	}
	public String toString() {
		return educationBackground + skills;
	}
}

/* 建立类,实现Clone方法  */
class Resume  implements Cloneable{
	private String name;  //姓名
	private String sex;   //性别
	private int age;      //年龄
	private Experience experience; //工作经历

	public Resume(String name, String sex, int age) {
		this.name = name;
		this.sex = sex;
		this.age = age;
		this.experience = new Experience();
	}

	public void setAge(int age) {
		this.age = age;
	}
	public int getAge() {
		return age;
	}

	public Experience getExperience() {
		return experience;
	}

	public void setExperience(String educationBackground, String skills) {
		experience.setExperience(educationBackground, skills);
	}

	public void displayResume() {
		System.out.println("姓名:"+name+" 性别:"+sex+" 年龄:"+age);
		System.out.println("工作经历:"+experience.toString());
	}

	public Object clone() {
		try {
			return (Resume)super.clone();
		} catch (Exception e) {
			e.printStackTrace();
			return null;
		}
	}
}

public class MainClass {
	public static void main(String[] args) {
		Resume zhangsan = new Resume("zhangsan","男",24);
		zhangsan.setExperience("2009-2013就读于家里蹲大学","精通JAVA,C,C++,C#等代码拷贝和粘贴");
		zhangsan.displayResume();

		Resume zhangsan2 = (Resume)zhangsan.clone();
		zhangsan2.setExperience("2009-2013就读于家里蹲大学","精通JAVA,C,C++,C#等");
		zhangsan2.displayResume();
		zhangsan.displayResume();
		zhangsan2.displayResume();
	}
}

程序运行结果:

姓名:zhangsan 性别:男 年龄:24
工作经历:2009-2013就读于家里蹲大学精通JAVA,C,C++,C#等代码拷贝和粘贴
姓名:zhangsan 性别:男 年龄:24
工作经历:2009-2013就读于家里蹲大学精通JAVA,C,C++,C#等
姓名:zhangsan 性别:男 年龄:24
工作经历:2009-2013就读于家里蹲大学精通JAVA,C,C++,C#等
姓名:zhangsan 性别:男 年龄:24
工作经历:2009-2013就读于家里蹲大学精通JAVA,C,C++,C#等

我们看一下上面两段程序差异在哪儿,第一段程序的工作经历是作为Resume类的一个普通的成员变量,也就是值属性。而后面一段程序中,工作经历Experience是一个类。结合上面程序的运行结果,我们再来理解“如果该字段是引用类型的话,则复制引用但不复制引用的对象。因此,原始对象及其副本引用同一个对象。”其实也就是说,zhangsan和zhangsan2里面的Experience类指向的是同一个对象嘛!那不管是zhangsan里面的Experience变化,还是zhangsan2里面的Experience变化都会影响另外一个啊。

浅拷贝,大家懂没?Over了啊!

3、深拷贝

由前面的分析,浅拷贝无法实现含有其他对象引用的本对象的拷贝。那么很显然,深拷贝,就是说创建一个新对象,然后将当前对象的非静态字段复制到该新对象,无论该字段是值类型的还是引用类型,都乖乖的进行复制。

有了这个出发点,其实改起来很好改啊。浅拷贝的死穴就在于原始对象及其副本引用同一个对象,那我们让他们不指向同一个对象不就完了嘛!见代码:

package interfaces.nesting;

class Experience {

	private String educationBackground;
	private String skills;

	public void setExperience(String educationBackground, String skills) {
		// TODO Auto-generated constructor stub
		this.educationBackground = educationBackground;
		this.skills = skills;
	}
	public String toString() {
		return educationBackground + skills;
	}
}

/* 建立类,实现Clone方法  */
class Resume  implements Cloneable{
	private String name;  //姓名
	private String sex;   //性别
	private int age;      //年龄
	private Experience experience; //工作经历

	public Resume(String name, String sex, int age) {
		this.name = name;
		this.sex = sex;
		this.age = age;
		this.experience = new Experience();
	}

	public void setAge(int age) {
		this.age = age;
	}
	public int getAge() {
		return age;
	}

	public Experience getExperience() {
		return experience;
	}

	public void setExperience(String educationBackground, String skills) {
		experience = new Experience();
		experience.setExperience(educationBackground, skills);
	}

	public void displayResume() {
		System.out.println("姓名:"+name+" 性别:"+sex+" 年龄:"+age);
		System.out.println("工作经历:"+experience.toString());
	}

	public Object clone() {
		try {
			return (Resume)super.clone();
		} catch (Exception e) {
			e.printStackTrace();
			return null;
		}
	}
}

public class MainClass {
	public static void main(String[] args) {
		Resume zhangsan = new Resume("zhangsan","男",24);
		zhangsan.setExperience("2009-2013就读于家里蹲大学","精通JAVA,C,C++,C#等代码拷贝和粘贴");
		zhangsan.displayResume();

		Resume zhangsan2 = (Resume)zhangsan.clone();
		zhangsan2.setExperience("2009-2013就读于家里蹲大学","精通JAVA,C,C++,C#等");
		zhangsan2.displayResume();
		zhangsan.displayResume();
		zhangsan2.displayResume();
	}
}

程序运行结果:

姓名:zhangsan 性别:男 年龄:24
工作经历:2009-2013就读于家里蹲大学精通JAVA,C,C++,C#等代码拷贝和粘贴
姓名:zhangsan 性别:男 年龄:24
工作经历:2009-2013就读于家里蹲大学精通JAVA,C,C++,C#等
姓名:zhangsan 性别:男 年龄:24
工作经历:2009-2013就读于家里蹲大学精通JAVA,C,C++,C#等代码拷贝和粘贴
姓名:zhangsan 性别:男 年龄:24
工作经历:2009-2013就读于家里蹲大学精通JAVA,C,C++,C#等

如有不足之处,欢迎指正啊!最后,祝大家睡个好觉(⊙o⊙)哦!See you~~

时间: 2024-10-13 11:59:17

一看就懂的,java深拷贝浅拷贝的相关文章

Java 深拷贝浅拷贝 与 序列化

一.浅拷贝.深拷贝 浅拷贝会对对象中的成员变量进行拷贝:如果是基本类型,拷贝的就是基本类型的值:如果属性是内存地址(引用类型),拷贝的就是内存地址 : 深拷贝,除了基本类型外,引用类型所引用的对象也会进行拷贝:(引用的对象只要求浅拷贝即可:若要深层拷贝,一般可利用序列化和反序列化来实现,也可手动实现各级引用对象的深层拷贝:) 二.实现: 浅拷贝:实现Cloneable接口,重写clone()方法,在clone()调用父类super.clone()即可: 深拷贝:1. 实现Cloneable接口,

java 深拷贝与浅拷贝机制详解

 java 深拷贝与浅拷贝机制详解            --摘自-https://www.jb51.net/article/106088.htm 概要: 在Java中,拷贝分为深拷贝和浅拷贝两种.java在公共超类Object中实现了一种叫做clone的方法,这种方法clone出来的新对象为浅拷贝,而通过自己定义的clone方法为深拷贝. (一)Object中clone方法 如果我们new出一个新对象,用一个声明去引用它,之后又用另一个声明去引用前一个声明,那么最后的结果是:这两个声明的变量将

一看就懂的Android APP开发入门教程

一看就懂的Android APP开发入门教程 作者: 字体:[增加 减小] 类型:转载 这篇文章主要介绍了Android APP开发入门教程,从SDK下载.开发环境搭建.代码编写.APP打包等步骤一一讲解,非常简明的一个Android APP开发入门教程,需要的朋友可以参考下 工作中有做过手机App项目,前端和android或ios程序员配合完成整个项目的开发,开发过程中与ios程序配合基本没什么问题,而android各种机子和rom的问题很多,这也让我产生了学习android和ios程序开发的

深拷贝&浅拷贝

STRING.h文件 #pragma once #include<string.h> class String { public: String(char* str="")      //深拷贝 :_str(new char[strlen(str)+1]) { strcpy(_str, str); cout << "构造函数 " << endl; } ~String() { if (_str!=NULL) { delete[]_s

探讨一下iOS中深拷贝&amp;浅拷贝&amp;copy的那些事儿

什么是深拷贝?什么是浅拷贝? 为什么经常看到字符串属性要这样定义,那个copy是神马意思? @property(nonatomic,copy)NSString* name; 为什么下面的写法是错误的? @property(nonatomic,copy)NSMutableString* name; copyWithZone方法又到底是干嘛用的? 接下来,我们将一起,一步一步的去揭晓问题的答案. Copy到底是个啥? 其实我们真的没必要把copy想的太高深.它之所以叫copy,其终极目的已不言而喻,

深拷贝 浅拷贝 引用计数

;深拷贝 浅拷贝 深拷贝(成员用到了指针存储空间地址)每个对象的成员都有自己独立的成员内存地址空间,造成了浪费 浅拷贝,把原对象的指针也直接拷贝过来我还是用的这一片空间,但是析构会有重复释放问题,解决重复释放用引用计数记录这个类产生了多少对象,析构的时候--引用计数就可以了 计数为0   delete 这块内存空间 ;引用计数技术 优点:所有对象共享同一片空间,间接的达到了对象间的数据共享 缺点:一旦一个对象改变了这片内存,那么所有的对象都受到影响 ;写时拷贝技术 当对象需要操作这一块空间存放数

深拷贝 浅拷贝 以及赋值运算符= 的重载

在一些程序当中,需要将一个对象里的值直接传递给另一个对象,需要进行对象的拷贝.但是在某些情况下也会出现错误,如类内部成员需要动态开辟内存,实行位拷贝,就是将一个对象的值完全复制给另一个对象,如A=B,但如果B中也有一个指针申请了内存,那么A中的成员变量也指向了同一块内存,如果当释放资源的时候,就会导致野指针的出现,出现错误. 深拷贝的简单理解 就是在复制对象内容的过程中,需要重新分配资源,而浅拷贝则是不要重新分配资源.但是浅拷贝存在着问题:当资源释放时会产生资源归属不清的问题,导致运行错误. 当

c# 通用类型系统 深拷贝 浅拷贝 函数传参

c# 通用类型系统 及变量在 深拷贝 浅拷贝 函数传参 中的深层次的表现 在编程中遇到了一些想不到的异常,跟踪发现,自己对于c#变量在内存上的表现理解有偏差,系统的学习并通过代码实验梳理了各种情况下,变量在内存级的表现情况,对以后的coding应该有些帮助.在此记录以免忘记了... 1. 通用类型系统 先来一张图: 通用数据类型分为了值类型和引用类型. 我们定义一个int型实际上是一个system.int32的实例,在语法上我们像使用其他的类对象一样,但是,存储的的仍然是基本类型.这种把基本类型

看得懂的 Node.js(三)—— Express 启航

如果看过上一篇<看得懂的 Node.js>,就会发现手动搭建一个 web 服务器还是比较繁琐 而 express 就是一个可以极大地提高开发效率的 web 开发框架 一.创建项目 在 express 4.0 之前,我们使用 npm install -g express 来全局安装 express 但是 4.0 之后,express 的命令行工具被单独分离出来,叫做 express-generator npm install -g express-generator 如果了解过 vue,expr