免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 1342 | 回复: 0
打印 上一主题 下一主题

java中的clone [复制链接]

论坛徽章:
1
数据库技术版块每日发帖之星
日期:2015-06-12 22:20:00
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2015-06-11 09:59 |只看该作者 |倒序浏览
1.先说一句
不去覆盖clone方法,不去调用clone方法,除非真的有必要。

2.java中clone两个概念
浅克隆 copy的是引用
深克隆 copy的是实例,开辟新的堆空间

java中的clone方法实现的是浅克隆,一个类可被浅克隆需实现Cloneable(此接口只是说明此类允许clone,它改变的是超类中受保护的clone方法的行为),如effective java中的第11条所说这是接口的一种极端非典型的用法,不值得效仿。
3.浅克隆
浅克隆看上去只要实现了Cloneable接口 且@Override clone方法为public后就可以了,但是实际应用了,确定你要的是浅clone么?
若是一个对象没有实现Cloneable接口,也可以很简单的使用反射实现对象的浅复制:
非严谨的code如下:
  1. public static <T> T simpleClone(T obj) throws IllegalAccessException, InstantiationException {
  2.         Class<T> c = (Class<T>) obj.getClass();
  3.         T cloneC = c.newInstance();
  4.         Field[] fields = c.getDeclaredFields();
  5.         if (fields != null && fields.length > 0) {
  6.             for (Field field : fields) {
  7.                 field.setAccessible(true);
  8.                 Object value = field.get(obj);
  9.                 field.set(cloneC, value);
  10.             }
  11.         }
  12.         return cloneC;
  13.     }
复制代码
当然有很多工具类了:比如,spring的BeanUtils.copyProperties, apache的BeanUtils.copyProperties,cglib或者spring-cglib的BeanCopier

4.深克隆
既然实际应用中更希望使用的是深克隆,那么如何实现呢
bean 自己实现Cloneable接口,靠谱的实现clone方法
非严谨code如下:
  1. public class TestCloneBean implements Cloneable {
  2.    
  3.     private List<Integer> integers;

  4.     public List<Integer> getIntegers() {
  5.         return integers;
  6.     }

  7.     public void setIntegers(List<Integer> integers) {
  8.         this.integers = integers;
  9.     }

  10.     @Override
  11.     public TestCloneBean clone() {
  12.         try {
  13.             TestCloneBean t = (TestCloneBean) super.clone();
  14.             t.setIntegers(Lists.<Integer> newArrayList());
  15.             if (CollectionUtils.isNotEmpty(integers)) {
  16.                 for (Integer i : integers) {
  17.                     t.getIntegers().add(i);
  18.                 }
  19.             }
  20.             return t;

  21.         } catch (CloneNotSupportedException e) {
  22.             throw new RuntimeException(e);
  23.         }
  24.     }

  25.     @Override
  26.     public String toString() {
  27.         return ToStringBuilder.reflectionToString(this);
  28.     }
  29. }
复制代码
第一种方法你会这么做么?不会。那么只能另外一种方式:序列化啊!
非严谨code如下:
  1. public class JsonUtils {
  2.     private static ObjectMapper objectMapper = new ObjectMapper();

  3.     static {
  4.         objectMapper.configure(JsonParser.Feature.ALLOW_COMMENTS, true);
  5.         objectMapper.configure(JsonParser.Feature.ALLOW_UNQUOTED_FIELD_NAMES, true);
  6.         objectMapper.configure(JsonParser.Feature.ALLOW_SINGLE_QUOTES, true);
  7.         objectMapper.configure(JsonParser.Feature.ALLOW_UNQUOTED_CONTROL_CHARS, true);
  8.         objectMapper.configure(JsonParser.Feature.INTERN_FIELD_NAMES, true);
  9.         objectMapper.configure(JsonParser.Feature.CANONICALIZE_FIELD_NAMES, true);
  10.         objectMapper.configure(DeserializationConfig.Feature.FAIL_ON_UNKNOWN_PROPERTIES, false);
  11.         objectMapper.setSerializationInclusion(Inclusion.NON_NULL);
  12.     }

  13.     public static ObjectMapper getObjectMapperInstance() {
  14.         return objectMapper;
  15.     }

  16. }
复制代码
testCode如下:
  1. TestCloneBean b = new TestCloneBean();
  2.         b.setIntegers(Lists.newArrayList(1));
  3.         String s = JsonUtils.getObjectMapperInstance().writeValueAsString(b);
  4.         TestCloneBean a = JsonUtils.getObjectMapperInstance().readValue(s, TestCloneBean.class);
复制代码
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP