zoukankan      html  css  js  c++  java
  • 设计模式-原型模式

    1、原型模式简介

    定义:属于创建型模式,用一个已经创建的实例作为原型,通过复制该原型对象来创建一个和原型相同或相似的新对象,这种方式创建对象非常高效,根本无须知道对象创建的细节

    目标:用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象

    解决问题:运行期建立和删除实例

    优点:性能提高

        逃避构造函数的约束

    缺点:需要为每一个类都配置一个 clone 方法

       clone 方法位于类的内部,当对已有类进行改造的时候,需要修改代码,违背了开闭原则。

       当实现深克隆时,需要编写较为复杂的代码,而且当对象之间存在多重嵌套引用时,为了实现深克隆,每一层对象对应的类都必须支持深克隆,实现起来会比较麻烦

    说明:原型模式的克隆分为两种,一种叫深克隆,一种叫浅克隆

    2、浅克隆

    定义:创建一个新对象,新对象的属性和原来对象完全相同,对于非基本类型属性,仍指向原有属性所指向的对象的内存地址

    实现:必须得实现实现 Cloneable 接口,只是复制了对象的引用地址,两个对象指向同一个内存地址,所以修改其中任意的值,另一个值都会随之变化

    //原型类
    public class User {
        private String name;
        private int date;
        private int age;
        public void setName(String name) {
            this.name = name;
        }
        public void setDate(int date) {
            this.date = date;
        }
        public void setAge(int age) {
            this.age = age;
        }
    
        public String getName() {
            return name;
        }
    
        public int getDate() {
            return date;
        }
    
        public int getAge() {
            return age;
        }
    
        @Override
        public String toString() {
            return getName()+"&&"+getDate()+"&&"+getAge();
        }
    }
    
    //具体原型类
    public class CloneablePrototype implements Cloneable {
        private User user = new User();
        public CloneablePrototype(){
            user.setAge(25);
            user.setDate(1995);
            user.setName("CC");
            System.out.println("具体原型创建成功");
        }
        public Object clone() throws CloneNotSupportedException {
            System.out.println("具体原型复制成功");
            return (CloneablePrototype)super.clone();
        }
        public User show(){
            return user;
        }
    
    }
    
    //测试方法
    public static void main(String[] args) throws CloneNotSupportedException {
            CloneablePrototype cloneablePrototype1 = new CloneablePrototype();
            CloneablePrototype cloneablePrototype2 = (CloneablePrototype) cloneablePrototype1.clone();
            System.out.println("clone1:"+cloneablePrototype1.show().toString());
            System.out.println("clone2:"+cloneablePrototype2.show().toString());
            System.out.println(cloneablePrototype1 == cloneablePrototype2);
            System.out.println("===========================");
            cloneablePrototype2.show().setAge(30);
            System.out.println("clone1:"+cloneablePrototype1.show().toString());
            System.out.println("clone2:"+cloneablePrototype2.show().toString());
            System.out.println(cloneablePrototype1 == cloneablePrototype2);
    
    }
    //测试结果
    具体原型创建成功
    具体原型复制成功
    clone1:CC&&1995&&25
    clone2:CC&&1995&&25
    false
    ===========================
    clone1:CC&&1995&&30
    clone2:CC&&1995&&30
    false

    3、深克隆

    定义:创建一个新对象,属性中引用的其他对象也会被克隆,不再指向原有对象地址

    实现:实现 Serializable 接口,通过字节流来对数据进行复制

    //原型类,实现Serializable接口
    public class User extends Object implements Serializable {
        private static final long serialVersionUID = 1L;
        private String name;
        private int date;
        private int age;
        public void setName(String name) {
            this.name = name;
        }
        public void setDate(int date) {
            this.date = date;
        }
        public void setAge(int age) {
            this.age = age;
        }
    
        public String getName() {
            return name;
        }
    
        public int getDate() {
            return date;
        }
    
        public int getAge() {
            return age;
        }
    
        @Override
        public String toString() {
            return getName()+"&&"+getDate()+"&&"+getAge();
        }
    }
    
    //具体原型类
    public class SerializablePrototype implements Serializable {
    
        private User user = new User();
        public SerializablePrototype(){
            user.setAge(21);
            user.setName("NN");
            user.setDate(1998);
            System.out.println("具体原型创建成功");
        }
        public User show(){
            return user;
        }
        public SerializablePrototype sClone(SerializablePrototype serializablePrototype){
            try(ByteArrayOutputStream bOut = new ByteArrayOutputStream();
                ObjectOutputStream oOut = new ObjectOutputStream(bOut)){
                //oOut对象中指向写入数据存储的内存地址与bOut对象中指向数据存储的内存地址是一样的
                oOut.writeObject(serializablePrototype);
                try(ByteArrayInputStream bIn= new ByteArrayInputStream(bOut.toByteArray());
                    ObjectInputStream oIn = new ObjectInputStream(bIn)){
                    Object o = oIn.readObject();
                    System.out.println("具体原型复制成功");
                    return (SerializablePrototype) o;
                }catch (IOException e) {
                    e.printStackTrace();
                } catch (ClassNotFoundException e) {
                    e.printStackTrace();
                }
            } catch (IOException e) {
                e.printStackTrace();
            }
            return  null;
        }
    }
    
    //测试方法
    public static void main(String[] args) throws CloneNotSupportedException {
            SerializablePrototype s1 = new SerializablePrototype();
            SerializablePrototype s2 = s1.sClone(s1);
            System.out.println(s1.show().toString());
            System.out.println(s2.show().toString());
            System.out.println(s1 == s2);
            System.out.println("============================");
            s2.show().setName("QQ");
            System.out.println(s1.show().toString());
            System.out.println(s2.show().toString());
            System.out.println(s1 == s2);
    }
    //测试结果
    具体原型创建成功
    具体原型复制成功
    NN&&1998&&21
    NN&&1998&&21
    false
    ============================
    NN&&1998&&21
    QQ&&1998&&21
    false
    
    由此可见,s2改变之后s1的不变,说明通过字节流将s1对象指向的实际内存地址中的数据复制到另一块实际内存中,s2指向的这块新的内存地址
  • 相关阅读:
    WCF BasicHttpBinding 安全解析(2)BasicHttpBinding安全项
    WCF BasicHttpBinding 安全解析(4)windows验证(IIS宿主)
    网页优化系列三:使用压缩后置viewstate
    向IIS请求页面时,它做了什么?!(部分转载)
    使用Odbc链接mysql时parameter参数设置
    .net中close和dispose及关闭流操作
    asp.net 解码gb2312下urlencode后的字符串
    GridView实战一:自定义分页、排序、修改、插入、删除
    实现滑动分页(微博分页方式)
    javascript实例:逐条记录停顿的走马灯
  • 原文地址:https://www.cnblogs.com/carblack/p/13749579.html
Copyright © 2011-2022 走看看