java 对象的克隆(浅克隆和深克隆)
作者:沈君 发布时间:2021-06-27 20:43:20
标签:java,对象,克隆
java 对象的克隆
一、对象的浅克隆
(1)需要克隆类需要重写Object类的clone方法,并且实现Cloneable接口(标识接口,无需实现任何方法)
(2)当需要克隆的对象中维护着另外一个引用对象,浅克隆不会克隆另外一个引用对下,而是直接复制维护的另外一个引用对象的地址。
(3)对象的浅克隆也不会调用到构造方法。
以下为对象的浅克隆的一个例子:
package com.clone;
import java.io.Serializable;
/**
* Description:
* 实现了Cloneable接口,并重写Object类的clone方法。
*
* @author lee
* */
public class CloneDemo1 implements Cloneable,Serializable{
//该克隆类封装的信息
public int id;
public String name;
public Address address;
/**
* Desciption:
* 默认构造器
*
* */
public CloneDemo1(){}
/**
* Description:
* 初始化id,name的构造器
*
* @param id id
* @param name 名字
* @param address 地址
* */
public CloneDemo1(int id, String name, Address address){
this.id=id;
this.name=name;
this.address = address;
}
/**
* Descriptin:
* 重写Object类的clone方法。
* if the object's class does not support the Cloneable interface.
* Subclasses that override the clone method can also throw this exception
* to indicate that an instance cannot be cloned.
*
* @throws CloneNotSupportedException
* */
@Override
public Object clone() throws CloneNotSupportedException{
return super.clone();
}
/**
* Description:
* 重写toString方法
*
* @return "id="+id+", name="+name
* */
@Override
public String toString(){
return "id="+id+", name="+name+", address:"+address.getAddress();
}
/**
* Description:
* 主方法
*
* */
public static void main(String[] args) throws CloneNotSupportedException{
CloneDemo1 c1 = new CloneDemo1(1,"c1",new Address("北京"));
//c2 复制了c1的地址,并没有复制整个c1对象
CloneDemo1 c2 = c1;
//c3 对象的浅克隆,复制了整个对象
CloneDemo1 c3 = (CloneDemo1)c1.clone();
//当对象c1改变其name或者id的时候,c2也会自动改变。
//因为c2只是复制了c1的地址,并非复制了c1的整个对象。
//相应的c3则不会随着c1改变而改变,意味着c3将c1整个对象克隆一份出来。
//当是,对象的浅克隆不会克隆被克隆对象当中的引用对象。
//因此c1改变其中的Address的引用对象时,c2,c3也会跟着改变。
c1.setName("cc");
c1.address.setAddress("上海");
System.out.println(c1+"\n"+c2+"\n"+c3);
}
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
/**
* Description:
* 一个封装着地址的类
*
* @author lee
* */
class Address implements Serializable{
public String address;
/**
* Description:
* 默认构造器
*
* */
public Address(){}
/**
* Description:
* 初试化address
*
* @param address 地址
* */
public Address(String address){
this.address = address;
}
//address的set和get方法
public String getAddress() {
return address;
}
public void setAddress(String address) {
this.address = address;
}
}
二、对象的深克隆
就是利用对象的输入输出流把对象写到文件上,再读取对象的信息,这就是对象的深克隆。
由于对象的浅克隆不会克隆被克隆对象其中的引用对象,而是直接复制其地址。因此,要克隆被克隆对象当中的引用类型则需要对象的深克隆。
而对象的深克隆使用的的对象序列化输入输出。
代码如下:
package com.clone;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
/**
* Description:
* 实现对象的深克隆
*
* @author lee
* */
public class CloneDemo2 {
/**
* Description:
* 将对象输出到一个文件当中。
*
* @param c 需要被写到文件当中的对象。
* */
public static void writeObject(CloneDemo1 c){
ObjectOutputStream out = null;
try{
//将对象输出在一个object.txt文件当中
out = new ObjectOutputStream(new FileOutputStream("./object.txt"));
out.writeObject(c);
}catch(IOException e){
System.out.println("写入对象的时候发生了错误。");
e.printStackTrace();
}finally{
//关闭资源
try{
out.close();
}catch(IOException e){
e.printStackTrace();
}
}
}
/**
* Description:
* 从文件中读取出一个对象来,并返回。
*
* @return c 返回一个对象。
* */
public static CloneDemo1 readObject(){
CloneDemo1 c = null;
ObjectInputStream input = null;
try{
//从object.txt文件中读取一个对象出来
input = new ObjectInputStream(new FileInputStream("./object.txt"));
c = (CloneDemo1)input.readObject();
}catch(IOException | ClassNotFoundException e){
e.printStackTrace();
System.out.println("读取对象的时候发生了错误。");
}finally{
//关闭资源
try{
input.close();
}catch(IOException e){
e.printStackTrace();
}
}
return c;
}
/**
* Description:
* 主方法
*
* @throws CloneNotSupportedException
* */
public static void main(String[] args) throws CloneNotSupportedException {
CloneDemo1 c1 = new CloneDemo1(1,"c1",new Address("北京"));
//c2 对象的浅克隆
CloneDemo1 c2 = (CloneDemo1)c1.clone();
//c3对象的深克隆
writeObject(c1);
CloneDemo1 c3 = readObject();
//因为对象的深克隆同时也克隆了被克隆对象维护的另外一个对象
//所以,当c1改变其当中的维护的另外一个对象的时候,c3不会随之改变。
//而c2位浅克隆,其维护的另外一个对象只是复制了c1维护的对象的地址,因此会随着c1的改变而改变。
c1.address.setAddress("上海");
System.out.println(c1+"\n"+c2+"\n"+c3);
}
}
对象的序列化,是需要实现Serializable接口的。
感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!
来源:http://blog.csdn.net/qq1131410679/article/details/65934698


猜你喜欢
- 线程安全当多个线程访问某个类时,不管运行时环境采用何种调度方式或者这些进程将如何交替执行,并且在主调代码中不需要任何额外的同步或协调,这个类
- 很多方法请看官网地址:hutool官网地址脚本之家下载:https://www.jb51.net/softs/549331.html简介Hu
- 本文实例为大家分享了springmvc实现验证码功能展示的具体代码,供大家参考,具体内容如下先看效果图: 思路: 首先验证码是一张图片,是一
- 本文介绍通过Java程序批量替换PDF中的指定文本内容。程序环境准备如下:程序使用环境如图,需要注意的是,本文使用了免费版的PDF jar工
- 前言日常开发中常见的文件格式有pdf,word,Excel,PPT,Html,txt,图片等。pdf,Html,txt,图片这种实现在线预览
- 本文实例讲述了Android编程实现启动界面的方法。分享给大家供大家参考,具体如下:最近在弄一个程序启动界面程序,在这里贴下代码。解释一下:
- *res/raw和assets的相同点:1.两者目录下的文件在打包后会原封不动的保存在apk包中,不会被编译成二进制。*res/raw和as
- 关联篇:HandlerThread 使用及其源码完全解析关联篇:Handler内存泄漏详解及其解决方案一说到Android的消息机制,自然就
- socketpair()函数的声明:#include <sys/types.h>#include <sys/socket.
- 前言在 Java 中,跳转的实现方式有两种:请求转发和请求重定向,但二者是完全不同的,所以我们今天就来盘它。请求转发和请求重定向主要区别,包
- 初始化项目打开IntelliJ IDEA,我的版本是Version 2018.1.4。点击Create New Project。在左侧的列表
- 1.修改系统默认的Dialog样式(风格、主题)2.自定义Dialog布局文件3.可以自己封装一个类,继承自Dialog或者直接使用Dial
- spring与IoCIoC:控制反转,将由代码操纵的对象控制权,交给第三方容器,反转给第三方容器。这种对象依赖的关系管理方式,称作IoC。I
- Java单例模式实现的几种方式单例模式好多书上都是这么写的:public class SingleTon1 {private SingleT
- 目录认识@Import注解搭建项目结构用于测试@Import用法最佳搭档 - @Import通用形式总结认识@Import注解先看一下源码@
- 本文实例主要实现了网页照相机程序的功能。C#实现将网页保存成图片格式,简单实现网页拍照,主要是基于ActiveX 组件的网页快照类,Acit
- 一、为什么要学习并发编程对于 “我们为什么要学习并发编程?” 这个问题,就好比 “我们为什么要学习政治?” 一样,我们(至少作为学生党是这样
- 1 SharedPreferences 介绍SharedPreferences是使用键值对的方式来存储数据的SharedPreference
- Java 中的运算符与 C 语言基本一致。1、算术运算符操作符描述例子+加法 : 相加运算符两侧的值A + B 等于 30-减法 : 左操作
- Android设备的内存有限,对于大图片,必须进行压缩后再进行显示,否则会出现内存溢出:OOM;处理策略:1.使用缩略图(Thumbnail