zoukankan      html  css  js  c++  java
  • java基础---->Java关于复制的使用(一)

      这里简单记录一下java中关于浅复制和深复制的知识。很多时候,一个人选择了行走,不是因为欲望,也并非诱惑,他仅仅是听到了自己内心的声音。

    java中的复制clone方法

    一、java对象的浅复制

      一个实现了Cloneable并重写了clone方法的类A,有一个无参构造或有参构造B,通过new关键字产生了一个对象S,再然后通过S.clone()方式产生了一个新的对象T,那么在对象拷贝时构造函数B是不会被执行的

    public class Thing implements Cloneable {
        //定义一个私有变量
        private ArrayList<String> arrayList = new ArrayList<String>();
    
        public Thing() {
            System.out.println("构造函数被执行了......");
        }
    
        @Override
        protected Thing clone() {
            Thing thing = null;
            try {
                thing = (Thing) super.clone();
            } catch (CloneNotSupportedException e) {
                e.printStackTrace();
            }
            return thing;
        }
    
        //设置HashMap的值
        public void setValue(String value) {
            this.arrayList.add(value);
        }
    
        //取得arrayList的值
        public List<String> getValue() {
            return this.arrayList;
        }
    }

      Thing类实现了Cloneable接口,并重写了Object的clone方法,里面有一个ArrayList列表的私有变量。现在我们的测试代码如下:

    package com.linux.huhx.learn.clone;
    
    /**
     * @Author: huhx
     * @Date: 2017-12-26 上午 10:57
     */
    public class ClientMain {
        public static void main(String[] args) {
            // 产生一个对象
            Thing thing = new Thing();
            thing.setValue("huhx");
    
            // 复制一个对象
            Thing cloneThing = thing.clone();
            cloneThing.setValue("linux");
            System.out.println(thing.getValue());
        }
    }

    运行的结果如下:

    构造函数被执行了......
    [huhx, linux]

      从结果我们可以看到:thing.clone()执行并没有再次调用构造函数,复制后的对象和原对象共享私有变量arrayList,也就是说复制对象对arrayList的操作会映射到原对象的arrayList。这样的浅拷贝是有风险的,下面我们介绍一下怎样实现对象的深复制。我们只需要在Thing类中clone方法加上数组的复制代码就可以,clone方法的代码如下:

    @Override
    protected Thing clone() {
        Thing thing = null;
        try {
            thing = (Thing) super.clone();
            thing.arrayList = (ArrayList<String>) this.arrayList.clone();
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        return thing;
    }

      现在再执行ClientMain的main方法,打印的结果如下:可以看到原对象的arrayList并没有改变。

    构造函数被执行了......
    [huhx]

      需要注意的是:对象的clone与对象内的final关键字是有冲突的,我们修改Thing类的私有变量arrayList的修饰符。

    private final ArrayList<String> arrayList = new ArrayList<String>();

      通过ide我们可以看到在thing.arrayList = (ArrayList<String>) this.arrayList.clone();代码处会有编译报错。所以要想实现对象的深复制,类的成员变量上不要增加final关键字。

    Cannot assign a value to final variable 'arrayList'

    友情链接

  • 相关阅读:
    Solr4.10.2集成Nutch1.9与自带UI界面使用
    Solr4.10.2的IK Analyzer分词器配置
    Solr4.10.2的Tomcat配置
    Nutch1.9安装配置与基本使用介绍
    ubuntu14.10中tomcat8设置管理员帐号
    ubuntu14.10折腾tomcat8,是的,折腾
    spring webservice 开发demo (实现基本的CRUD 数据库采用H2)
    spring webservice 搭建出现的异常处理。异常: NAMESPACE_ERR: An attempt is made to create or change an object in a way whi
    hibernate.cfg.xml hibernate 配置文件模板
    spring 与 CXF 整合 webservice 出现error “Unable to locate Spring NamespaceHandler for XML schema namespace” 总结
  • 原文地址:https://www.cnblogs.com/huhx/p/baseusejavaclone1.html
Copyright © 2011-2022 走看看