JAVA Clone复制对象
谈到了对象的克隆,就不得不说为什么要对对象进行克隆。Java中所有的对象都是保存在堆中,而堆是供全局共享的。也就是说,如果同一个Java程序的不同方法,只要能拿到某个对象的引用,引用者就可以随意的修改对象的内部数据(前提是这个对象的内部数据通过get/set方法曝露出来)。有的时候,我们编写的代码想让调用者只获得该对象的一个拷贝(也就是一个内容完全相同的对象,但是在内存中存在两个这样的对象),有什么办法可以做到呢?当然是克隆咯。
import java.io.Serializable; public class Apple implements Cloneable,Serializable {
private static final long serialVersionUID = 1L; private String locality; public String getLocality() {
return locality;
} public void setLocality(String locality) {
this.locality = locality;
} @Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
}
上面的代码,简单的构建了一个Apple对象,并实现了Cloneable及Serializable接口,并重写clone()方法。
接着再构建一个Fruit对象,里面有一个Apple属性。
import java.io.Serializable; public class Fruit implements Cloneable,Serializable{
private static final long serialVersionUID = 1L; private String fruitName; private Apple apple; public Apple getApple() {
return apple;
} public void setApple(Apple apple) {
this.apple = apple;
} public String getFruitName() {
return fruitName;
} public void setFruitName(String fruitName) {
this.fruitName = fruitName;
} @Override
protected Fruit clone() throws CloneNotSupportedException { return (Fruit)super.clone();
}
}
接着复制一下对象:
Fruit fruit = new Fruit();
fruit.setFruitName("Apple");
Apple apple = new Apple();
apple.setLocality("China");
fruit.setApple(apple); Fruit fruit2 = fruit.clone(); System.out.println("Fruit = "+(fruit==fruit2));
System.out.println("Fruit's apple="+(fruit.getApple()==fruit2.getApple()));
Fruit = false
Fruit's apple=true
很明显,这里的复制是浅复制(shallow clone)。fruit2内的apple属性与fruit内的apple属性是同一个对象,只复制了引用的地址。
而深克隆,就是非浅克隆。克隆除自身以外所有的对象,包括自身所包含的所有对象实例。至于深克隆的层次,由具体的需求决定,也有“N层克隆”一说。
但是,所有的基本(primitive)类型数据,无论是浅克隆还是深克隆,都会进行原值克隆。毕竟他们都不是对象,不是存储在堆中。注意:基本数据类型并不包括他们对应的包装类。
如果我们想让对象进行深度克隆,我们可以这样修改Fruit类的clone()方法。
@Override
protected Fruit clone() throws CloneNotSupportedException {
Fruit fruit = (Fruit)super.clone();
fruit.apple = (Apple)fruit.getApple().clone();
return fruit; }
这样就可以实现了,如果类层次较深,这样写就很麻烦。
接下来要重点介绍一下使用java.lang.Serializable来实现对象的深度克隆。
首先,我们编写一个工具类并提供cloneTo()方法。
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream; public class BeanDeepCloneUtil { @SuppressWarnings("unchecked")
public static <T> T cloneTo(T src) {
ByteArrayOutputStream memoryBuffer = new ByteArrayOutputStream();
ObjectOutputStream outputStream = null;
ObjectInputStream inputStream = null; T destination = null;
try {
outputStream = new ObjectOutputStream(memoryBuffer);
outputStream.writeObject(src);
outputStream.flush();
inputStream = new ObjectInputStream(new ByteArrayInputStream(
memoryBuffer.toByteArray())); destination = (T) inputStream.readObject();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (ClassNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} finally { try {
if (outputStream != null) {
outputStream.close();
outputStream=null;
}
} catch (IOException e) {
e.printStackTrace();
} try {
if (inputStream != null) {
inputStream.close();
inputStream=null;
}
} catch (IOException e) {
e.printStackTrace();
}
} return destination;
} }
接下来我们测试一下是否能通过这个工具来实现深度克隆。
Apple apple = new Apple();
apple.setLocality("China"); Fruit fruit = new Fruit();
fruit.setFruitName("Apple");
fruit.setApple(apple); Fruit fruit3 = BeanDeepCloneUtil.cloneTo(fruit);
System.out.println("Fruit = "+(fruit==fruit3));
System.out.println("Fruit's apple="+(fruit.getApple()==fruit3.getApple()));
运行结果如下:
Fruit = false
Fruit's apple=false
JAVA Clone复制对象的更多相关文章
- java中复制对象通过反射或序列化
在使用缓存读取数据后修改发现缓存被修改.于是找了下复制对象的方法. 关于对象克隆 按我的理解,对象是包含引用+数据.通常变量复制都是将引用传递过去.比如: Person p1 = new Person ...
- Java提高篇——对象克隆(复制)
假如说你想复制一个简单变量.很简单: int apples = 5; int pears = apples; 不仅仅是int类型,其它七种原始数据类型(boolean,char,byte,short, ...
- Java 深复制和浅复制
浅复制是指复制对象时仅仅复制对象本身(包括对象中的基本变量),而不复制对象包含的引用指向的对象.深复制不仅复制对象本身,而且复制对象包含的引用指向的对象. 复制对象时需要调用Object类的clone ...
- 速战速决 (5) - PHP: 动态地创建属性和方法, 对象的复制, 对象的比较, 加载指定的文件, 自动加载类文件, 命名空间
[源码下载] 速战速决 (5) - PHP: 动态地创建属性和方法, 对象的复制, 对象的比较, 加载指定的文件, 自动加载类文件, 命名空间 作者:webabcd 介绍速战速决 之 PHP 动态地创 ...
- java复制对象之深拷背
在java开发中,有时我们需要复制对象,并且确保修改复制得到的对象不会影响原来的对象. 于是,有些人可能会写出类似以下的代码: public class CloneTest { public stat ...
- Java 数组复制之clone方法
一.源码 public class Test1 { public static void main(String[] args) { // Student[] arrs = new Student[] ...
- Java clone() 方法克隆对象——深拷贝与浅拷贝
基本数据类型引用数据类型特点 1.基本数据类型的特点:直接存储在栈(stack)中的数据 2.引用数据类型的特点:存储的是该对象在栈中引用,真实的数据存放在堆内存里 引用数据类型在栈中存储了指针,该指 ...
- Java反射 - 2(对象复制,父类域,内省)
为什么要复制对象?假设有个类Car,包含name,color2个属性,那么将car1对象复制给car2对象,只需要car2.setName(car1.getName)与car2.setColor(ca ...
- [置顶] ruby复制对象的方法(dup 和 clone)
Ruby内置的方法Object#clone和Object#dup可以用来copy一个对象,两者区别是dup只复制对象的内容,而clone还复制与对象相关联的内容,如singleton method [ ...
随机推荐
- 2017-4-20/Redis的数据结构及应用场景
1. 谈谈你对redis的理解,它的应用场景. Redis是一个key-value存储系统,它支持存储的value类型包括string字符串.list链表.set集合.sorted Set有序集合和h ...
- Spring Boot系列之配置日志输出等级
我们都知道Spring boot 默认使用 logback作进行日志输出,那么 在配置Spring boot日志输出时有两种方式: 通过application.properties 配置文件的方式来配 ...
- 1.Liunx 文件管理
文件管理命令 1.基本命令:ls .cd .pwd .man 2.文件操作:touch . cp . rm . mv .ln . mkdir . rmdir 3.文件查看:cat .more .les ...
- Oracle物理DG自动切换——Dataguard Broker配置
1.前言部分 1.1 导读 各位技术爱好者,看完本文后,你可以掌握如下的技能: ① Data Guard Broker 的配置 ② Fast-Start Failover 的配置 ③ Oracle D ...
- [LeetCode] 201. Bitwise AND of Numbers Range ☆☆☆(数字范围按位与)
https://leetcode.com/problems/bitwise-and-of-numbers-range/discuss/56729/Bit-operation-solution(JAVA ...
- ActiveMQ 事务和XA
1. 客户端怎样显式地使用事务? producer 开启事务(代码片段): ActiveMQSession session = (ActiveMQSession)connection.createSe ...
- redis的文件事件
redis的文件事件:即与io相关的事件. /* File event structure */ typedef struct aeFileEvent { int mask; /* one of AE ...
- !important 的绝对控制样式
<head> <style type="text/css"> div{background-color: blue !important;} </st ...
- mysql 如何在访问某张数据表按照某个字段分类输出
也许大家有时候会遇到需要将把数据库中的某张表的数据按照该表的某个字段分类输出,比如一张数据表area如下 我们需要将里面的area按照serialize字段进行分类输出,比如这种形式: areas ...
- std::string find 的返回值
std::string 的方法 find,返回值类型是std::string::size_type, 对应的是查找对象在字符串中的位置(从0开始), 如果未查找到,该返回值是一个很大的数据(4294 ...