zoukankan      html  css  js  c++  java
  • Chapter 9 原型模式

    原型模式:用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象. 简单的说就是clone一个对象实例。使得clone出来的copy和原有的对象一模一样。 插一个简单使用clone的例子,如果一个对象内部有可变对象实例的话,public API不应该直接返回该对象的引用,以防调用方的code改变该对象的内部状态。这个时候可以返回该对象的clone。 一般而言,我们要的clone应该是这样的。copy和原型的内容一样,但是又是彼此隔离的。即在clone之后,改变其中一个不影响另外一个。 

    原型模式分为浅复制和深复制,其中浅复制:被复制对象的所有变量都含有与原来的对象相同的值,而所有的对其他对象的引用都仍然指向原来的对象。深复制把引用对象的变量指向复制过的新对象,而不是原有的被引用的对象。

    class User {
    	String name;
    	int age;
    }
    class Account implements Cloneable {
    	User user;
    	long balance;
    	@Override
    	public Object clone() throws CloneNotSupportedException {
    		return super.clone();
    	}
    }

    我们之所以要深copy,是因为默认的实现提供的浅copy不是隔离的,换言之,改变copy的东西,会影响到原型的内部。比如例子中,改变copy的user的name,影响了原型。特别注意的是string是一种拥有值类型特点的特殊引用类型。
    class User implements Cloneable {
    	String name;
    	int age;
    	@Override
    	public User clone() throws CloneNotSupportedException {
    		return (User) super.clone();
    	}
    }
    class Account implements Cloneable {
    	User user;
    	long balance;
    	@Override
    	public Account clone() throws CloneNotSupportedException {
    		Account account = null;
    		account = (Account) super.clone();
    		if (user != null) {
    			account.user = user.clone();
    		}
    		return account;
    	}
    }

    再来贴2个实例代码

    一、浅复制:

    package xiao;

     

    class WorkExp{

    private String workDate;

    private String company;

    public String getWorkDate() {

    return workDate;

    }

    public void setWorkDate(String workDate) {

    this.workDate = workDate;

    }

    public String getCompany() {

    return company;

    }

    public void setCompany(String company) {

    this.company = company;

    }

    }

    class Resume implements Cloneable{

    private String name;

    private String sex;

    private String age;

    private WorkExp work;

    public Resume(String name){

    this.name = name;

    work = new WorkExp();

    }

    public void setPerInfo(String sex,String age){

    this.sex = sex;

    this.age = age;

    }

    public void setWorkExper(String timeArea,String company){

    work.setWorkDate(timeArea);

    work.setCompany(company);

    }

    public void display(){

    System.out.println(this.name+" "+this.sex+" "+this.age);

    System.out.println(work.getWorkDate()+" "+work.getCompany());

    }

    public Object clone() throws CloneNotSupportedException {

    return super.clone();

    }

    }

    public class Hello {

     

    public static void main(String[] args) throws Exception{

    Resume a = new Resume("big bird");

    a.setPerInfo("male", "20");

    a.setWorkExper("1998-2006", "IT company");

    Resume b = (Resume)a.clone();

    b.setWorkExper("1998-2004", "YY company");

    Resume c = (Resume)a.clone();

    c.setPerInfo("female", "22");

    c.setWorkExper("1998-2004", "ZZ company");

    a.display();

    b.display();

    c.display();

    }

    }

    输出:

    big bird male 20
    1998-2006 ZZ company
    big bird male 20
    1998-2004 ZZ company
    big bird female 22
    1998-2004 ZZ company

    二、深复制:

    
    

    package xiao;

    
    

    class WorkExp implements Cloneable{
    private String workDate;
    private String company;
    public String getWorkDate() {
    return workDate;
    }
    public void setWorkDate(String workDate) {
    this.workDate = workDate;
    }
    public String getCompany() {
    return company;
    }
    public void setCompany(String company) {
    this.company = company;
    }
    public WorkExp clone() throws CloneNotSupportedException {
    return (WorkExp)super.clone();
    }
    }
    class Resume implements Cloneable{
    private String name;
    private String sex;
    private String age;
    private WorkExp work;
    public Resume(String name){
    this.name = name;
    work = new WorkExp();
    }
    public void setPerInfo(String sex,String age){
    this.sex = sex;
    this.age = age;
    }
    public void setWorkExper(String timeArea,String company){
    work.setWorkDate(timeArea);
    work.setCompany(company);
    }
    public void display(){
    System.out.println(this.name+" "+this.sex+" "+this.age);
    System.out.println(work.getWorkDate()+" "+work.getCompany());
    }
    public Resume clone() throws CloneNotSupportedException {
    Resume resume = null;
    resume = (Resume) super.clone();
    if(work != null){
    resume.work = work.clone();
    }
    return resume;
    }
    }
    public class Hello {

    
    

    public static void main(String[] args) throws Exception{
    Resume a = new Resume("big bird");
    a.setPerInfo("male", "20");
    a.setWorkExper("1998-2006", "IT company");
    Resume b = (Resume)a.clone();
    b.setWorkExper("1998-2004", "YY company");
    Resume c = (Resume)a.clone();
    c.setPerInfo("female", "22");
    c.setWorkExper("1998-2004", "ZZ company");
    a.display();
    b.display();
    c.display();
    }
    }

    输出:

    big bird male 20
    1998-2006 IT company
    big bird male 20
    1998-2004 YY company
    big bird female 22
    1998-2004 ZZ company

  • 相关阅读:
    Objective-C ,ios,iphone开发基础:ios数据库(The SQLite Database),使用终端进行简单的数据库操作
    Objective-C ,ios,iphone开发基础:picker控件详解与使用,(实现省市的二级联动)
    Objective-C ,ios,iphone开发基础:多个视图(view)之间的切换2,使用导航栏控制,以及视图之间传值。
    Objective-C ,ios,iphone开发基础:多个视图(view)之间的切换,以及视图之间传值。
    Objective-C ,ios,iphone开发基础:NSDictionary(字典) 和 NSMutableDictionary
    Objective-C ,ios,iphone开发基础:几个常用类-NSString
    为MYPoint类写一个分类
    使用copy再次实现Circle类,保证不能有内存泄漏问题
    实现Square类,让其继承自Rectangle类,并在Square类增添新属性和方法,在2的基础上,在Square类中重写Rectangle类中的初始化和打印方法
    实现Square类,让其继承自Rectangle类,并在Square类增添新属性和方法
  • 原文地址:https://www.cnblogs.com/tuifeideyouran/p/3733905.html
Copyright © 2011-2022 走看看