原型模式——五种创建型模式之一

 
 

1.前言


单例模式可以避免重复创建消耗资源的对象,但是却不得不共用对象。若是对象本身也不让随意访问修改时,怎么办?通常做法是备份到副本,其它对象操作副本,最后获取权限合并,类似git上的PR操作。

2.概念


原型模式用原型实例指定创建对象的种类,并通过拷贝这些原型创建新的对象。需要注意的关键字是,新的对象,类没变。Java正好提供了Cloneable接口,它标识的类可以调用Object中实现的clone()方法而不抛出异常,即运行时通知虚拟机可以安全使用clone()方法返回拷贝对象。由于它直接操作内存中的二进制流,当大量操作或操作复杂对象时,性能优势将会很明显。

3.场景


动物园中有一只羊,对它进行克隆,产生另外一只完全一样的羊,分别安排两位有孩子的管理员照顾。有一天,对克隆羊进行基因操作,观察变化。

4.写法

// 1.声明此类可以被clone
public class Sheep implements Cloneable {
    
    private int age;
    private String sex;
    private Admin admin;

    public Sheep(int age, String sex, Admin admin) {
        super();
        this.age = age;
        this.sex = sex;
        this.admin = admin;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    public String getSex() {
        return sex;
    }

    public void setSex(String sex) {
        this.sex = sex;
    }

    public Admin getAdmin() {
        return admin;
    }

    public void setAdmin(Admin admin) {
        this.admin = admin;
    }

    @Override
    public String toString() {
        return "Sheep [age=" + age + ", sex=" + sex + ", admin=" + admin + "]";
    }
    
    // 2.调用Object的clone方法
    public Sheep startClone() {
        Sheep sheep = null;
        try {
            sheep = (Sheep) super.clone();
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        return sheep;
    }

}
public class Admin {
    
    private int age;
    private String sex;
    private Child child;
    public Admin(int age, String sex, Child child) {
        super();
        this.age = age;
        this.sex = sex;
        this.child = child;
    }
    
    public void setAge(int age) {
        this.age = age;
    }

    public void setSex(String sex) {
        this.sex = sex;
    }

    public void setChild(Child child) {
        this.child = child;
    }

    @Override
    public String toString() {
        return "Admin [age=" + age + ", sex=" + sex + ", child=" + child + "]";
    }
    
}
public class Child {

}
public class Zoo {

    public static void main(String[] args) {
        Sheep old = new Sheep(2, "雄性", new Admin(25, "女", new Child()));
        System.out.println(old.toString());
        Sheep current = old.startClone();
        System.out.println(current.toString());
        
        // 对克隆羊做处理
        current.setAge(1);
        current.setSex("雌性");
        current.getAdmin().setAge(34);
        current.getAdmin().setSex("男");
        System.out.println(old.toString());
        System.out.println(current.toString());
    }

}

根据上面的代码,我们知道羊引用了管理员,管理员引用了孩子。当对内存中数据拷贝时,除了基本数据类型(包括封装类型)及String类型,其它的引用关系将直接传递给副本,并不是重新创建一个对象。所以当对克隆羊操作时,年龄和性别直接改变,而对管理员的操作将寻址到内存中对应部分进行修改,导致原型也被修改。孩子与管理员的关系就如同管理员与羊,通过哈希值可以知道,孩子始终就一个,没有拷贝成功。


light clone.png

上面的错误是由于只拷贝了最外层对象的原因,我们称之为浅拷贝。为了解决这个问题,需要对内部的引用类型进行拷贝(Java中大部分引用类型实现了Cloneable接口,可以方便的拷贝),具体如下:

// 1.声明此类可以被clone
public class Sheep implements Cloneable {

    // 前面省略
    
    // 2.调用Object的clone方法
    public Sheep startClone() {
        Sheep sheep = null;
        try {
            sheep = (Sheep) super.clone();
            
            // 3.调用Admin的clone方法
            sheep.admin = this.admin.startClone();
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        return sheep;
    }

}
public class Admin implements Cloneable {

    // 前面省略
    
    public Admin startClone() {
        Admin admin = null;
        try {
            admin = (Admin) super.clone();
            admin.child = this.child.startClone();
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        return admin;
    }
    
}
public class Child implements Cloneable {

    public Child startClone() {
        Child child = null;
        try {
            child = (Child) super.clone();
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        return child;
    }
    
}

通过日志的打印,发现这种方式(深拷贝)起作用了。由1、2行可以知道,拷贝时引用类型已经重新创建了对象。由3、4行可以知道,修改其中一个对象不会再改变另一个了。


deep clone.png

5.总结


原型模式通过Object的clone()方法实现,由于是内存操作,无视构造方法和访问权限,直接获取新的对象。但对于引用类型,需使用深拷贝,其它浅拷贝即可。

猜你喜欢

转载自blog.csdn.net/imimi_/article/details/79247052