对象浅克隆问题及解决方案

前天车险批改进行发票拆分打印测试时,引出了一个克隆问题,现补分个享,就当作知识巩固了。
应用场景:发票拆分,为对一个单某期次下拆分成多条金额数据,即拆分后会有多个Contract对象,其中Contract含有List< PayInfoDTO>(支付信息)对象。然而在程序中,根据单号连库只执行查询一次加载Contrat,因金额数据是由外部呈多条传入,有几条金额数据便有几个Contract,并set到Contrac的List< PayInfoDTO>对象中。
问题:因克隆时,只对Contract作浅克隆,即实现Cloneable接口,导致取Contract中的List<PayInfo>对象永远是最后一次set金额,这里其实是一个对象引用问题。
解决方案:在Contract的clone中实现了只对List< PayInfoDTO>的深克隆(网上说这是一个淹菜>回鲜的过程),代码如:

public Object clone() {
ContractDTO o = null;
try {
o = (ContractDTO) super.clone();
List<PayInfoDTO> payInfoList = o.getPayInfoList();
if (CommonFunctions.isCollectionNotEmpty(payInfoList)) {
List<PayInfoDTO> bakPayInfoList = new ArrayList<PayInfoDTO>();
for (PayInfoDTO payInfo : payInfoList) {
bakPayInfoList.add(deepClone(payInfo));
}

o.setPayInfoList(bakPayInfoList);
}
} catch (CloneNotSupportedException e) {
} catch (IOException e) {
} catch (ClassNotFoundException e) {
}
return o;
}

public static <T> T deepClone(T src) throws IOException, ClassNotFoundException {
T t = null;
ObjectOutputStream oos = null;
ObjectInputStream ois = null;
try {
ByteArrayOutputStream baos = new ByteArrayOutputStream();
oos = new ObjectOutputStream(baos);
oos.writeObject(src);
oos.flush();
ois = new ObjectInputStream(new ByteArrayInputStream(baos.toByteArray()));
t = (T) ois.readObject();
} finally {
if (ois != null) {
ois.close();
}
if (oos != null) {
oos.close();
}
}
return t;
}

代码详见:com.paic.icore.acss.print.biz.service.impl.PrintPojoService.getInvoiceInfoForSplit(CancelAndPrintReqForm cancelAndPrintReqForm)
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值