• 设计模式学习(五):原型模式


    设计模式学习(五):原型模式

    作者:Grey

    原文地址:

    博客园:设计模式学习(五):原型模式

    CSDN:设计模式学习(五):原型模式

    原型模式

    原型模式是创建型模式。

    如果对象的创建成本比较大,而同一个类的不同对象之间差别不大(大部分字段的值都相同),在这种情况下,我们可以利用对已有对象(原型)进行复制(或者叫拷贝)的方式来创建新对象,以达到节省创建时间的目的。这种基于原型来创建对象的方式就叫作原型设计模式。

    实际上,创建对象包含的申请内存、给成员变量赋值这一过程,本身并不会花费太多时间,或者说对于大部分业务系统来说,这点时间完全是可以忽略的。应用一个复杂的模式,只得到一点点的性能提升,这就是所谓的过度设计,得不偿失。但是,如果对象中的数据需要经过复杂的计算才能得到(比如排序、计算哈希值),或者需要从 RPC 、网络、数据库、文件系统等非常慢速的 I/O 中读取,每次读取一次的代价都很高,在这种情况下,我们就可以利用原型模式,从其他已有对象中直接拷贝得到,而不用每次在创建新对象的时候,都重复执行这些耗时的操作。

    原型模式用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象,典型的应用是对象的克隆方法,示例代码如下

    public class Person implements Cloneable {
        String name = "lisa";
        int age = 1;
        Location loc = new Location("xy", 10);
    
        @Override
        protected Object clone() throws CloneNotSupportedException {
            Person p = (Person) super.clone();
            p.loc = (Location) loc.clone();
            return p;
        }
    
        @Override
        public String toString() {
            return "Person{" + "name='" + name + '\'' + ", age=" + age + ", loc=" + loc + '}';
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    public class Location implements Cloneable {
        private String street;
        private int roomNo;
    
        public Location(String street, int roomNo) {
            this.street = street;
            this.roomNo = roomNo;
        }
    
        @Override
        protected Object clone() throws CloneNotSupportedException {
            return super.clone();
        }
    
        @Override
        public String toString() {
            return "Location{" + "street='" + street + '\'' + ", roomNo=" + roomNo + '}';
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    public class Main {
        public static void main(String[] args) throws CloneNotSupportedException {
            Person p = new Person();
            System.out.println(p);
            Person p2 = (Person) p.clone();
            System.out.println(p2);
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8

    本示例的 UML 图如下:

    image

    注:Java 自带的 clone() 方法进行的就是浅克隆。而如果我们想进行深克隆,可以直接在类中调用父类的克隆方法,即: super.clone() 后,手动给克隆对象的相关属性分配另一块内存,不过如果当原型对象维护很多引用属性的时候,手动分配会比较烦琐。因此,在 Java 中,如果想完成原型对象的深克隆,则通常使用序列化的方式。

    例如:克隆一个巨大的 HashMap,如果构建该 HashMap 的代价很大,我们可以通过

    方式一:先调用 HashMap 默认的克隆方法,然后递归拷贝 HashMap 里面的内容,直到类型是基础类型为止;

    方式二:使用序列化方式克隆。

    关于序列化克隆的示例代码如下

    import java.io.*;
    import java.util.ArrayList;
    import java.util.List;
    
    public class DeepCloneDemo {
        public static void main(String[] args) {
            List<String> hobbies = new ArrayList<>();
            hobbies.add("唱歌");
            hobbies.add("跳舞");
            Prototype p = new Prototype();
            p.setAge(18);
            p.setName("zhangsan");
            p.setHobbits(hobbies);
            Prototype clone = p.clone();
            System.out.println(clone.getAge());
            System.out.println(clone.getName());
            System.out.println(clone.getHobbits());
        }
    }
    
    // 待克隆对象
    class Prototype implements Cloneable, Serializable {
        private int age;
        private String name;
        private List<String> hobbits;
    
        public int getAge() {
            return age;
        }
    
        public void setAge(int age) {
            this.age = age;
        }
    
        public String getName() {
            return name;
        }
    
        public void setName(String name) {
            this.name = name;
        }
    
        public List<String> getHobbits() {
            return hobbits;
        }
    
        public void setHobbits(List<String> hobbits) {
            this.hobbits = hobbits;
        }
    
        @Override
        protected Prototype clone() {
            return deepClone();
        }
    
        public Prototype deepClone() {
            try {
                ByteArrayOutputStream bos = new ByteArrayOutputStream();
                ObjectOutputStream oos = new ObjectOutputStream(bos);
                oos.writeObject(this);
                ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray());
                ObjectInputStream ois = new ObjectInputStream(bis);
                return (Prototype) ois.readObject();
            } catch (Exception e) {
                e.printStackTrace();
                return null;
            }
        }
    
        @Override
        public String toString() {
            return "Prototype{" + "age=" + age + ", name='" + name + '\'' + ", hobbits=" + hobbits + '}';
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74

    如果只是增量拷贝,可以通过浅拷贝拿到一个新的 HashMap ,然后拿到增量的数据单独进行深拷贝即可。

    更多地,Spring 中创建对象的方式默认采用单例模式,可以通过设置 @Scope("prototype") 注解将其改为原型模式。

    UML 和 代码

    UML 图

    代码

    更多

    设计模式学习专栏

    参考资料

  • 相关阅读:
    Taurus.MVC WebMVC 入门开发教程4:数据列表绑定List<Model>
    关于电话号码欺骗的一切
    测绘屠夫报表系统V1.0.0-beta
    Intel base instruction -- cpuid
    阿里云数据库大全_3分钟看懂阿里云RDS和NoSQL数据库汇总
    云原生技术 --- k8s配置组件之ConfigMap的学习与使用
    Go-知识测试-模糊测试
    1783_CMD启动MATLAB同时执行一个脚本
    Unity中Shader的ShadowMapping的原理(阴影)
    信息反馈平台的设计与实现(一、项目设计)
  • 原文地址:https://blog.csdn.net/hotonyhui/article/details/127750096