前天车险批改进行发票拆分打印测试时,引出了一个克隆问题,现补分个享,就当作知识巩固了。
应用场景:发票拆分,为对一个单某期次下拆分成多条金额数据,即拆分后会有多个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)
应用场景:发票拆分,为对一个单某期次下拆分成多条金额数据,即拆分后会有多个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)