美文网首页
Android源码设计模式学习笔记-原型模式

Android源码设计模式学习笔记-原型模式

作者: e小e | 来源:发表于2018-01-07 10:22 被阅读11次

原型模式主要用来克隆一个对象,在开发过程中我们会有需要一个对象的copy这种需求,常常就会用克隆这种设计模式.
在java中需要克隆的对象必须要实现一个,不实现Cloneable接口也可以重写clone方法,但是调用clone方法会抛出CloneNotSupportedException

public interface Cloneable {
}

并且重写clone()方法.
例如我需要User具有clone的功能.

public class User implements Cloneable{
    public int age;
    public String name;

    public User(int age, String name) {
        this.age = age;
        this.name = name;
    }

    @Override
    public User clone() throws CloneNotSupportedException {
        return (User) super.clone();
    }
}

在clone一个User可以

User user = new User(25,"jay");
try {
     User temp = user.clone();
     System.out.println("name " +temp.name+" age "+temp.age);
} catch (CloneNotSupportedException e) {
     e.printStackTrace();
}

最后输出

System.out: name jay age 25

上面演示了一种简单浅克隆,下面再来谈谈深度克隆.
User.class现在变成了这样

public class User implements Cloneable{
    public int age;
    public String name;
    public ArrayList<String> friends;

    public User(int age, String name) {
        this.age = age;
        this.name = name;
        friends = new ArrayList<>();
    }

    public void add(String friend){
        friends.add(friend);
    }

    @Override
    public User clone() throws CloneNotSupportedException {
        return (User) super.clone();
    }
}

注意,它多了一个List<String> friends
再看看克隆的时候

User user = new User(25,"jay");
user.add("小明");
user.add("小红");
user.add("小白");
try {
     User temp = user.clone();
     temp.add("小黑");
     for (String f : temp.friends){
          System.out.println("copy "+f);
     }
} catch (CloneNotSupportedException e) {
     e.printStackTrace();
}
     for (String f : user.friends){
             System.out.println("original "+f);
     }
}

输出:

01-07 10:00:26.649 7331-7331/com.example.huangli.rankdemo I/System.out: copy 小明
01-07 10:00:26.649 7331-7331/com.example.huangli.rankdemo I/System.out: copy 小红
01-07 10:00:26.649 7331-7331/com.example.huangli.rankdemo I/System.out: copy 小白
01-07 10:00:26.649 7331-7331/com.example.huangli.rankdemo I/System.out: copy 小黑
01-07 10:00:26.649 7331-7331/com.example.huangli.rankdemo I/System.out: original 小明
01-07 10:00:26.649 7331-7331/com.example.huangli.rankdemo I/System.out: original 小红
01-07 10:00:26.649 7331-7331/com.example.huangli.rankdemo I/System.out: original 小白
01-07 10:00:26.649 7331-7331/com.example.huangli.rankdemo I/System.out: original 小黑

当你改变克隆内容的时候,原始数据也改变了,这显然不是我们想看到的.
User中的clone是可以写自己的克隆方式的

@Override
public User clone() throws CloneNotSupportedException {
     return (User) super.clone();
}

我们需要改写

@Override
public User clone() throws CloneNotSupportedException {
    User user = (User) super.clone();
    user.friends = (ArrayList<String>) this.friends.clone();
    return user;
}

ArrayList重写了clone方法实现了数组的copy.
最后运行一下

01-07 10:14:58.162 8060-8060/com.example.huangli.rankdemo I/System.out: copy 小明
01-07 10:14:58.162 8060-8060/com.example.huangli.rankdemo I/System.out: copy 小红
01-07 10:14:58.162 8060-8060/com.example.huangli.rankdemo I/System.out: copy 小白
01-07 10:14:58.162 8060-8060/com.example.huangli.rankdemo I/System.out: copy 小黑
01-07 10:14:58.162 8060-8060/com.example.huangli.rankdemo I/System.out: original 小明
01-07 10:14:58.162 8060-8060/com.example.huangli.rankdemo I/System.out: original 小红
01-07 10:14:58.162 8060-8060/com.example.huangli.rankdemo I/System.out: original 小白

这次在修改克隆数据的同时并没有改变原始数据.
总结:
浅拷贝可以用于一些对象只具有基本数据结构的情况下进行拷贝,深拷贝可以实现数据结构复杂的情况下的深度拷贝.
另外还有一个疑问是,我可以重新构造一个对象,并不需要克隆模式去实现对象的拷贝,其实构造一个对象也是可以的,但是这时候需要权衡,哪个开销更大,应该构造函数可能不仅仅实现了数据的初始化,它可能还会去做一些别的事情(比如说去触发另外一个模块的启动),但是克隆就不一样了,它仅仅是去拷贝一份对象数据.

相关文章

网友评论

      本文标题:Android源码设计模式学习笔记-原型模式

      本文链接:https://www.haomeiwen.com/subject/uybynxtx.html