java cloneable 用途_java中cloneable的使用「建议收藏」

java cloneable 用途_java中cloneable的使用「建议收藏」什么是java中的浅克隆和深克隆?浅克隆:克隆对象中的变量与之前对象的值相同,并且对象中的引用类型变量仍然指向原来对象引用类型变量的地址.深克隆:克隆对象中的变量与之前对象的值相同,并且对象中的引用类型变量指向了新的对象的引用变量的地址.要想实现克隆,只需定义的类声明下cloneable这个标记性接口,并且衍生重写Object类中就有的clone()方法即可.为什么类要首先声明cloneable标…

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE稳定放心使用

什么是java中的浅克隆和深克隆?

浅克隆:克隆对象中的变量与之前对象的值相同,并且对象中的引用类型变量仍然指向原来对象引用类型变量的地址.

深克隆:克隆对象中的变量与之前对象的值相同,并且对象中的引用类型变量指向了新的对象的引用变量的地址.

要想实现克隆,只需定义的类声明下cloneable这个标记性接口,并且衍生重写Object类中就有的clone()方法即可.

为什么类要首先声明cloneable标记接口,然后重写clone()方法?因为不声明cloneable调用clone()方法会抛出CloneNotSupportedException异常,源码如下:

protected Object clone() throws CloneNotSupportedException {

if (!(this instanceof Cloneable)) {

throw new CloneNotSupportedException(“Class ” + getClass().getName() +

” doesn’t implement Cloneable”);

}

return internalClone();

}

/*

* Native helper method for cloning.

*/

private native Object internalClone();

在上一节中讲了java中Serializable与Parcelable的使用序列化与反序列化的问题。事实上利用对象输出流对对象进行序列化,利用对象的输入流对对象进行反序列化也可以实现克隆,如果对象中依赖的其他对象的引用也实现了序列化(即引用类实现了serializable标记接口)就实现了深度克隆,否则实现了浅克隆.

实现了Serializable接口的Company

public class Company implements Serializable {//Serializable接口是空的,没有声明的方法及常量

private static final long serialVersionUID = 1L; //序列化标识

private String name;

private String address;

public String getName() {

return name;

}

public void setName(String name) {

this.name = name;

}

public String getAddress() {

return address;

}

public void setAddress(String address) {

this.address = address;

}

public Company(String name, String address) {

this.name = name;

this.address = address;

}

@Override

public String toString() {

return “company name is:”+name+”,address is:”+address;

}

}

获得实现了Serializable接口的克隆实例调用方法。

private T getCopyObj(T t) {

ByteArrayOutputStream byteArrayOutputStream = null;

ObjectOutputStream objectOutputStream = null;

ByteArrayInputStream byteArrayInputStream = null;

ObjectInputStream objectInputStream = null;

try {

byteArrayOutputStream = new ByteArrayOutputStream();

objectOutputStream = new ObjectOutputStream(byteArrayOutputStream);

objectOutputStream.writeObject(t);//序列化对象

objectOutputStream.flush();

byteArrayInputStream = new ByteArrayInputStream(byteArrayOutputStream.toByteArray());

objectInputStream = new ObjectInputStream(byteArrayInputStream);

T t1 = (T) objectInputStream.readObject();

return t1;

} catch (Exception e) {

e.printStackTrace();

} finally {

if (byteArrayOutputStream != null) {

try {

byteArrayOutputStream.close();

byteArrayOutputStream = null;

} catch (IOException e) {

e.printStackTrace();

}

}

if (objectOutputStream != null) {

try {

objectOutputStream.close();

objectOutputStream = null;

} catch (IOException e) {

e.printStackTrace();

}

}

if (byteArrayInputStream != null) {

try {

byteArrayInputStream.close();

byteArrayInputStream = null;

} catch (IOException e) {

e.printStackTrace();

}

}

if (objectInputStream != null) {

try {

objectInputStream.close();

objectInputStream = null;

} catch (IOException e) {

e.printStackTrace();

}

}

}

return null;

}

}

测试通过的testCase,说明通过Serializable的反序列化创建的是一个新的对象,不再是之前的对象了。

@Test

public void test() throws CloneNotSupportedException {

Company company=new Company(“百度”,”上地十街”);

Company copyCompany=getCopyObj(company);

copyCompany.setName(“腾讯”);

Assert.assertEquals(false,company==copyCompany);

Assert.assertEquals(true,company.getName().equals(“百度”));

Assert.assertEquals(true,copyCompany.getName().equals(“腾讯”));

}

实现了Clonable克隆的例子

public class People implements Cloneable {

private String name;

private int age;

public People(String name, int age) {

this.name = name;

this.age = age;

}

@Override

protected Object clone() throws CloneNotSupportedException {

return super.clone();

}

public String getName() {

return name;

}

public void setName(String name) {

this.name = name;

}

public int getAge() {

return age;

}

public void setAge(int age) {

this.age = age;

}

}

验证通过的case,表明了克隆出来的对象与原来的对象地址不一样,是一个新的对象,所以克隆对象中的name和age是新的.

@Test

public void test() throws CloneNotSupportedException {

People people = new People(“storm”, 30);

People clonePeople = (People) people.clone();

clonePeople.setName(“stormClone”);

clonePeople.setAge(29);

Assert.assertFalse(people == clonePeople);

System.out.println(“people name=” + people.getName());//people name=storm

System.out.println(“people age=” + people.getAge());//people age=30

System.out.println(“clonePeople name=” + clonePeople.getName());//clonePeople name=stormClone

System.out.println(“clonePeople age=” + clonePeople.getAge());//clonePeople age=29

}

使用cloneable实现浅克隆

public class Animal {

private String animalName;

public Animal(String animalName) {

this.animalName = animalName;

}

public String getAnimalName() {

return animalName;

}

public void setAnimalName(String animalName) {

this.animalName = animalName;

}

}

public class People implements Cloneable {

private String name;

private int age;

private Animal animal;//克隆对象中的引用型变量

public People(String name, int age,Animal animal) {

this.name = name;

this.age = age;

this.animal=animal;

}

@Override

protected Object clone() throws CloneNotSupportedException {

return super.clone();

}

public String getName() {

return name;

}

public void setName(String name) {

this.name = name;

}

public int getAge() {

return age;

}

public void setAge(int age) {

this.age = age;

}

public Animal getAnimal() {

return animal;

}

public void setAnimal(Animal animal) {

this.animal = animal;

}

}

验证通过的case,表明了克隆对象的引用型变量animal并未发生改变,也即使内存中的地址并未发生改变,所以对其name的更改会影响原对象与克隆对象的值.

@Test

public void test() throws CloneNotSupportedException {

Animal animal=new Animal(“cat”);

People people = new People(“storm”, 30,animal);

People clonePeople = (People) people.clone();

animal.setAnimalName(“dog”);

Assert.assertFalse(people == clonePeople);

Assert.assertTrue(people.getAnimal()==clonePeople.getAnimal());

Assert.assertTrue(people.getAnimal().getAnimalName().equals(“dog”));

Assert.assertTrue(clonePeople.getAnimal().getAnimalName().equals(“dog”));

}

使用cloneable实现深克隆(实现很简单只需要引用类型变量实现cloneable接口即可),相比浅克隆,只需做如下修改.

public class Animal implements Cloneable{

private String animalName;

public Animal(String animalName) {

this.animalName = animalName;

}

public String getAnimalName() {

return animalName;

}

public void setAnimalName(String animalName) {

this.animalName = animalName;

}

@Override

protected Object clone() throws CloneNotSupportedException {

return super.clone();

}

}

验证通过的case,表明了克隆对象的引用型变量animal发生改变,也即内存中的地址发生改变,所以对其name的更改不会影响克隆对象的值.同时说明了进行深克隆会把所有的引用类型都实现cloneable接口,如果克隆对象中的引用类型变量比较多的话,这牵涉的工作量就会比较大了,这时我们考虑使用上面实现Serializable实现克隆的方式,缺点是反复进行IO操作,内存开销大.

@Test

public void test() throws CloneNotSupportedException {

Animal animal=new Animal(“cat”);

People people = new People(“storm”, 30,animal);

People clonePeople = (People) people.clone();

Animal cloneAnimal=(Animal) animal.clone();

clonePeople.setAnimal(cloneAnimal);

animal.setAnimalName(“dog”);

Assert.assertFalse(people == clonePeople);

Assert.assertFalse(people.getAnimal()==clonePeople.getAnimal());

Assert.assertTrue(people.getAnimal().getAnimalName().equals(“dog”));

Assert.assertTrue(clonePeople.getAnimal().getAnimalName().equals(“cat”));

}

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/181903.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 大约SQL现场“这包括”与“包括在”字符串的写法

    大约SQL现场“这包括”与“包括在”字符串的写法

    2022年1月3日
    55
  • Django(9)url指定默认参数「建议收藏」

    Django(9)url指定默认参数「建议收藏」前言当我们访问网页的时候,后台返回列表中有n条数据,此时我们会使用分页,比如一页只展示10条,但是我们访问第一页的时候大多数情况下,都会给url一个默认值,访问的时候直接展示第一页数据案例我们的

    2022年8月7日
    3
  • 带通滤波器原理及其作用_带通滤波器的设计

    带通滤波器原理及其作用_带通滤波器的设计一个带通滤波器是一个只有在特定频段的频率传递信号衰减这一频段以外的所有信号的同时,其目的是的电路。在一个带通滤波器的重要参数,高,低截止频率(FH和f升),带宽(BW),中心频率fC,中心频率增益,选择性或Q基本上有两个带通滤波器,即广通带和窄的带通滤波器的类型号不幸的是,两者之间的分线没有设置。然而,一个带通滤波器的定义是很宽的带通图的优点或品质因数Q值小于10,而与Q带通滤波器&g…

    2022年9月6日
    3
  • vue深拷贝与浅拷贝_vue深拷贝与浅拷贝

    vue深拷贝与浅拷贝_vue深拷贝与浅拷贝先说说1.栈内存,它的存取速度比较快,仅此于寄存器,栈中的数据还可以共享。其缺点表现在,存在栈中的数据大小与生存期必须是确定的,缺乏灵活性。使用完毕会被“隐式”回收2.堆内存,这和数据类型就有关系了,举个简单的例子leta=1,leta=b,你复制a,b会开辟新的栈内存,动态地分配内存大小,可以“按需分配”,其生存期也不必事先告诉编译器,在使用完毕后,Java的垃圾收集器会自动收走这些不再使用的内存块。其缺点为,由于要在运动时才动态分配内存,相比于栈内存,它的存取速度较慢。然后修改a

    2022年9月30日
    0
  • htmla标签下划线去除_html超链接的下划线怎么去掉?a标签去下划线的方法都在这里…

    htmla标签下划线去除_html超链接的下划线怎么去掉?a标签去下划线的方法都在这里…本篇文章就是关于html超链接取消下划线的用法,教你如何快速的去掉HTML超链接下划线的方法,最后还有相关代码解释,下面就让我们一起看看这篇文章吧首先我们使用css的基础样式来做一个最简单的去下划线的方法:htmla超链接标签,默认有的浏览器显示有下划线,有的没有下划线,大多锚文本超链接A标签内字体是有下划线的,怎么去除超链接下划线?html超链接去除下划线怎么做?去掉去除超链接锚文本的…

    2022年6月3日
    38
  • php 中js跳转页面跳转页面,js跳转代码_PHP页面跳转 Js页面跳转代码[通俗易懂]

    php 中js跳转页面跳转页面,js跳转代码_PHP页面跳转 Js页面跳转代码[通俗易懂]摘要腾兴网为您分享:PHP页面跳转Js页面跳转代码,自动刷宝,中信金通,携程抢票,未来屋等软件知识,以及沃金汇,沃行讯通,securecrt.exe,我的世界变形金刚mod,一票通,农场小分队,手电筒,推币机游戏,善行天下,硬盘mhdd,googlekeep,文件批量更名,明星表情包,服装销售软件,进击的巨人日语等软件it资讯,欢迎关注腾兴网。第一部分:JavaScript跳转方法一:…

    2022年8月13日
    1

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号