原型模式

原型模式定义
 用原型实例指定创建对象的种类,并通过拷贝这些原型创建新的对象。
 
原型模式示例

public class Person implements Cloneable {
    private String name;
    private int age;
    private double height;
    private double weight;
    public Person(){}
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public int getAge() {
        return age;
    }
    public void setAge(int age) {
        this.age = age;
    }
    public double getHeight() {
        return height;
    }
    public void setHeight(double height) {
        this.height = height;
    }
    public double getWeight() {
        return weight;
    }
    public void setWeight(double weight) {
        this.weight = weight;
    }

    @Override
    public String toString() {
        return "Person[name:"+name+", age:"+age+", height:"+height+", weight:" + weight+"]";
    }

    @Override
    protected Object clone() throws CloneNotSupportedException {
        Person person = null;
        try {
            person = (Person) super.clone();
            person.name = this.name;
            person.age = this.age;
            person.height = this.height;
            person.weight = this.weight;
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        return person;
    }
}

 这是原型类的实现,需要Person类的副本的时候调用clone方法,会生成一个副本,可以在副本里面修改而不影响原始的实例对象。这就是是浅拷贝,所谓浅拷贝说白了就是把原对象所有的值和引用直接赋给新对象。不会运行构造方法,如果构造方法里面有处理,拷贝的时候需要在clone方法里面做特效处理。
 

public static void main(String[] args) throws CloneNotSupportedException {
        Person person = new Person();
        person.setName("李四");
        person.setAge(22);
        person.setHeight(1.56);
        person.setWeight(66);
        System.out.println(person);
        Person person2 = (Person) person.clone();//创建副本

        System.out.println(person2);
        person2.setName("张三");
        System.out.println(person2);
        person.setName("wangwu");
        System.out.println(person);
        System.out.println(person2);
    }

测试结果:

Person[name:李四, age:22, height:1.56, weight:66.0]
Person[name:李四, age:22, height:1.56, weight:66.0]
Person[name:张三, age:22, height:1.56, weight:66.0]
Person[name:wangwu, age:22, height:1.56, weight:66.0]
Person[name:张三, age:22, height:1.56, weight:66.0]
 根据结果可以看出原型类和副本类的值互相不影响(不是引用类型,如果是引用类型,由于是浅拷贝,传递给副本的是引用类型的地址,两个互相影响,这里Person类里面没有引用类型,结果没有演示出来)。

有浅拷贝就有深拷贝,深拷贝则不仅把原对象的值赋给新对象,而且会把原对象的引用对象也重新创建一遍再赋给新对象(深拷贝,是把引用类型重新创建出来,所以副本的引用类型是和原型类里面的引用类型不是一个,互相不影响)。

 示例:

public class Person {
    private String name;
    ....//省略了一些字段
    private List<String> list;
    //省略get/set方法
    public Object clone(){
    ....//省略
        if(list != null){ //深拷贝,重新创建引用对象
            list = new ArrayList<String>();
        }
    }
}

Android中原型模式的使用
下面通过Intent来分析源码中的原型模型:

  Uri uri = Uri.parse("smsto:0800000124");
  Intent shareIntent = new Intent(Intent.ACTION_SEND,uri);
  shareIntent.putExtra("sms_body","The SMS text");
  //克隆副本
  Intent intent = (Intent) shareIntent.clone();
  startActivity(intent);

上面的代码是创建一个发送短信的shareIntent,然后克隆给新的intent,通过intent跳转启动短信发送信息。
下面来看看Intent的源码怎么实现:

  @Override
    public Object clone() {
        return new Intent(this);
    }
   /**
     * Copy constructor.
     */
    public Intent(Intent o) {
        this.mAction = o.mAction;
        this.mData = o.mData;
        this.mType = o.mType;
        this.mPackage = o.mPackage;
        this.mComponent = o.mComponent;
        this.mFlags = o.mFlags;
        this.mContentUserHint = o.mContentUserHint;
        //深拷贝
        if (o.mCategories != null) {
            this.mCategories = new ArraySet<String>(o.mCategories);
        }
        if (o.mExtras != null) {
            this.mExtras = new Bundle(o.mExtras);
        }
        if (o.mSourceBounds != null) {
            this.mSourceBounds = new Rect(o.mSourceBounds);
        }
        if (o.mSelector != null) {
            this.mSelector = new Intent(o.mSelector);
        }
        if (o.mClipData != null) {
            this.mClipData = new ClipData(o.mClipData);
        }
    }

可以看到clone方法实际上没有调用super.clone()方法来实现,而是调用了new Intent(this).其实实现克隆clone()方法里面不一定非要super.clone()来创建对象,可以使用new创建。考虑使用clone还是new是需要根据构造对象的成本来决策。具体使用那一个根据效率是实现。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值