Android 设计模式 - 原型模式

作者: Android架构 | 来源:发表于2019-02-28 21:24 被阅读4次

    原型模式如何使用

    原型模式主要用来克隆一个对象,在开发过程中我们会有需要一个对象的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架构交流QQ群聊:513088520

    点击链接加入群聊【Android移动架构总群】:加入群聊

    获取免费学习视频,学习大纲另外还有像高级UI、性能优化、架构师课程、NDK、混合式开发(ReactNative+Weex)等Android高阶开发资料免费分享。

    相关文章

      网友评论

        本文标题:Android 设计模式 - 原型模式

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