第一部分说了传值的问题,今天接着来看clone的问题。
package dcr.study.test.clone;
import java.util.Date;
/*
* 在实际编程中我们会遇到一种问题,比如,我们有一个客户,他每次来订货,订单上的商品几乎都是一样的
* 这时每次去录入商品,显然很烦,这个时候,我要参照以往的单据去创建一张新的订单。然后做些细微的修
* 改,这样就不用每次去录这个客户的信息和商品。
*
* 这种情况,用简单的赋值语句,不太好。有很多方法去实现这个需求,但用clone应该是一个比较好的方式。
*
*
* 这里值得注意的是Obj必须实现Cloneable接口,否则在使用clone方法的时候,会报
java.lang.CloneNotSupportedException异常
这里我们发现Cloneable接口是一个抽象接口,是不包含任何方法的
public abstract interface java.lang.Cloneable {}
其实这个接口仅仅是一个标志,而且这个标志也仅仅是针对Object类中clone()方法的,
如果clone类没有实现Cloneable接口,并调用了Object的clone()方法
(也就是调用了super.Clone()方法),那么Object的clone()方法就会抛出
CloneNotSupportedException异常。
* */
public class OrderClone implements Cloneable{
private String buyer;
private Date date;
private int number;
public String getBuyer() {
return buyer;
}
public void setBuyer(String buyer) {
this.buyer = buyer;
}
public Date getDate() {
return date;
}
public void setDate(Date date) {
this.date = date;
}
public int getNumber() {
return number;
}
public void setNumber(int number) {
this.number = number;
}
public Object clone(){
OrderClone o = null;
try {
o = (OrderClone)super.clone();
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
return o;
}
}
package dcr.study.test.clone;
/*
* 没有实现克隆接口
* */
public class UnCloneA {
private int i;
public UnCloneA(int ii) {
i = ii;
}
public void doublevalue() {
i *= 2;
}
public String toString() {
return Integer.toString(i);
}
}
package dcr.study.test.clone;
public class CloneB implements Cloneable {
public int aInt;
public UnCloneA unCA = new UnCloneA(123);
public Object clone() {
CloneB o = null;
try {
o = (CloneB) super.clone();
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
return o;
}
}
package dcr.study.test.clone;
import java.sql.Date;
/*总结:
Clone()方法的使用比较简单,注意如下几点即可:
a. 什么时候使用shallow Clone,什么时候使用deep Clone,这个主要看具体对象的域是什么性质的,
基本型别还是reference variable
b. 调用Clone()方法的对象所属的类(Class)必须implements Clonable接口,否则在调用Clone方法的
时候会抛出CloneNotSupportedException。如果仔细看你会发现Cloneable接口只是个抽象接口
并没有任何方法,他只是个标记。
* */
public class OrderTest {
public static void main(String[] args) {
OrderClone o1= new OrderClone();
o1.setBuyer("dcriori");
o1.setDate(new Date(0));
o1.setNumber(10000);
OrderClone o2 = o1;
OrderClone o3 = (OrderClone) o1.clone();
o2.setBuyer("sagaris");
o2.setNumber(20000);
System.out.println("O1 \nBuyer:"+o1.getBuyer()+"\nDate:"
+ o1.getDate() + "\nNumber:" + o1.getNumber());
System.out.println("O2 \nBuyer:"+o2.getBuyer()+"\nDate:"
+ o2.getDate() + "\nNumber:" + o2.getNumber());
System.out.println("O3 \nBuyer:"+o3.getBuyer()+"\nDate:"
+ o3.getDate() + "\nNumber:" + o3.getNumber());
/*
* 运行结果:
* O1
Buyer:sagaris
Date:1970-01-01
Number:20000
O2
Buyer:sagaris
Date:1970-01-01
Number:20000
O3
Buyer:dcriori
Date:1970-01-01
Number:10000
通过运行结果我们发现 OrderClone o2 = o1;o2是通过赋值创建的对象,o2在改变的时候,把o1也改
掉了,这不是我们想要的结果。但o3没有变,说明用clone创建出来的对象,和原来的对象没指向同一块
内存。
* */
System.out.println("============================================");
System.out.println("=============下面是影子克隆的情况===============");
System.out.println("============================================");
CloneB b1 = new CloneB();
b1.aInt = 11;
System.out.println("before clone,b1.aInt = "+ b1.aInt);
System.out.println("before clone,b1.unCA = "+ b1.unCA);
CloneB b2 = (CloneB)b1.clone();
b2.aInt = 22;
b2.unCA.doublevalue();
System.out.println("=================================");
System.out.println("after clone,b1.aInt = "+ b1.aInt);
System.out.println("after clone,b1.unCA = "+ b1.unCA);
System.out.println("=================================");
System.out.println("after clone,b2.aInt = "+ b2.aInt);
System.out.println("after clone,b2.unCA = "+ b2.unCA);
/* ============================================
=============下面是影子克隆的情况===============
============================================
before clone,b1.aInt = 11
before clone,b1.unCA = 123
=================================
after clone,b1.aInt = 11
after clone,b1.unCA = 246
=================================
after clone,b2.aInt = 22
after clone,b2.unCA = 246
* 这里我们发现,类CloneB虽然实现了Cloneable接口,也写了Clone方法,但是由于使用了其它未
* 实现Cloneable接口的类UnCloneA因此在改变b2的unCA的值的时候,把b1的unCA的值也改掉了
* 这说明在clone的时候,对于基础类型没有什么问题,但是对于类类型,问题就来了,我们知道它们
* 保存的仅仅是对象的引用,这也导致clone后的非基本类型变量和原始对象中相应的变量指向的是同一个对象。
*
* 解决这个问题的办法,就是深度clone,
* 其实说起来也很简单,两个方法,
* 一是把UnCloneA类也实现Cloneable接口,重载clone()方法;
* 二是在CloneB的clone()方法中加入一句o.unCA = (UnCloneA)unCA.clone();
* 代码就不再写了。
* */
}
}
分享到:
相关推荐
python学习资源
jfinal-undertow 用于开发、部署由 jfinal 开发的 web 项目
基于Andorid的音乐播放器项目设计(国外开源)实现源码,主要针对计算机相关专业的正在做毕设的学生和需要项目实战练习的学习者,也可作为课程设计、期末大作业。
python学习资源
python学习资源
python学习一些项目和资源
【毕业设计】java-springboot+vue家具销售平台实现源码(完整前后端+mysql+说明文档+LunW).zip
HTML+CSS+JavaScarip开发的前端网页源代码
python学习资源
【毕业设计】java-springboot-vue健身房信息管理系统源码(完整前后端+mysql+说明文档+LunW).zip
成绩管理系统C/Go。大学生期末小作业,指针实现,C语言版本(ANSI C)和Go语言版本
1_基于大数据的智能菜品个性化推荐与点餐系统的设计与实现.docx
【毕业设计】java-springboot-vue交流互动平台实现源码(完整前后端+mysql+说明文档+LunW).zip
内容概要:本文主要探讨了在高并发情况下如何设计并优化火车票秒杀系统,确保系统的高性能与稳定性。通过对比分析三种库存管理模式(下单减库存、支付减库存、预扣库存),强调了预扣库存结合本地缓存及远程Redis统一库存的优势,同时介绍了如何利用Nginx的加权轮询策略、MQ消息队列异步处理等方式降低系统压力,保障交易完整性和数据一致性,防止超卖现象。 适用人群:具有一定互联网应用开发经验的研发人员和技术管理人员。 使用场景及目标:适用于电商、票务等行业需要处理大量瞬时并发请求的业务场景。其目标在于通过合理的架构规划,实现在高峰期保持平台的稳定运行,保证用户体验的同时最大化销售额。 其他说明:文中提及的技术细节如Epoll I/O多路复用模型以及分布式系统中的容错措施等内容,对于深入理解大规模并发系统的构建有着重要指导意义。
基于 OpenCV 和 PyTorch 的深度车牌识别
【毕业设计-java】springboot-vue教学资料管理系统实现源码(完整前后端+mysql+说明文档+LunW).zip
此数据集包含有关出租车行程的详细信息,包括乘客人数、行程距离、付款类型、车费金额和行程时长。它可用于各种数据分析和机器学习应用程序,例如票价预测和乘车模式分析。
把代码放到Word中,通过开发工具——Visual Basic——插入模块,粘贴在里在,把在硅基流动中申请的API放到VBA代码中。在Word中,选择一个问题,运行这个DeepSeekV3的宏就可以实现在线问答
【毕业设计】java-springboot+vue机动车号牌管理系统实现源码(完整前后端+mysql+说明文档+LunW).zip
【毕业设计】java-springboot-vue交通管理在线服务系统的开发源码(完整前后端+mysql+说明文档+LunW).zip