原型模式,用原型实例指定建立对象的种类,而且经过拷贝这些原型建立新的对象。设计模式
Prototype原型模式是一种建立型设计模式,Prototype模式容许一个对象再建立另一个可定制的对象,根本无需知道任何如何建立的细节,工做原理是:经过将一个原型对象传给那个要发动建立的对象,这个要发动建立的对象经过请求原型对象拷贝它们本身来实施建立。微信
模型:ide
经过模型发现,就是经过Clone()方法来实现对对象的拷贝。函数
好比某平台举行抽奖送福利的活动,中奖的通知以邮件形式发送。思考一下,这里须要一个邮件Mail类用来填写姓名,内容等信息。性能
下面实例说明,先写普通Mail类和调用,而后再写出原型模式的。学习
1.没有实现Clone()方法的Mail类:this
public class Mail { //姓名 private String name; //地址 private String address; //内容 private String content; public Mail(){ Log.e("qzs","构造方法"); } public String getName() { return name; } public void setName(String name) { this.name = name; } public String getAddress() { return address; } public void setAddress(String address) { this.address = address; } public String getContent() { return content; } public void setContent(String content) { this.content = content; } }
2.调用spa
private int MAIL_COUNT=10; Mail mail=new Mail(); for (int i=0;i<MAIL_COUNT;i++){ mail.setName("姓名"+i); mail.setAddress("www.xxx"+i+".com"); mail.setContent("xxxxx"); sendMail(mail); } public void sendMail (Mail mail){ Log.e("qzs",mail.getName()+"邮件发送成功了"); }
运行:设计
考虑一个问题,须要发送邮件的数量不少的话,会很耗时。这时能够用咱们今天的讲的原型模式,来“克隆”出属性一致的对象来实现。code
下面修改了一个Mail类:
public class Mail implements Cloneable{ //姓名 private String name; //地址 private String address; //内容 private String content; public Mail(){ Log.e("qzs","构造方法"); } public String getName() { return name; } public void setName(String name) { this.name = name; } public String getAddress() { return address; } public void setAddress(String address) { this.address = address; } public String getContent() { return content; } public void setContent(String content) { this.content = content; } @Override protected Mail clone() throws CloneNotSupportedException { Log.e("qzs","clone"); Mail mail=null; try { mail= (Mail) super.clone(); } catch (CloneNotSupportedException e) { e.printStackTrace(); } return mail; } }
调用:
try { Mail mail=new Mail(); Mail mail_clone=mail.clone(); for (int i=0;i<MAIL_COUNT;i++){ mail_clone.setName("姓名"+i); mail_clone.setAddress("www.xxx"+i+".com"); mail_clone.setContent("xxxxx"); sendMail(mail_clone); } } catch (CloneNotSupportedException e) { e.printStackTrace(); } public void sendMail (Mail mail){ Log.e("qzs",mail.getName()+"邮件发送成功了"); }
运行:
原型模式是在内存中二进制流的拷贝,要比直接new一个对象性能好不少,尤为是构造函数比较复杂,而且在循环体中生产出大量的对象时,用原型模式效率很高。
原型模式-在拷贝时构造函数是不执行的。(上面的例子中已经发现了)
浅拷贝:在经过Clone() 方法实现拷贝对象时,只是对对象的拷贝,而没有对原生对象的其余因素进行拷贝,因此拷贝出的对象还指向原生的。
深拷贝:这就好理解了,拷贝对象和原生对象是分开的。
下面经过一个例子来讲明一下:
public class Text implements Cloneable{ private ArrayList<String> list=new ArrayList<>(); @Override protected Text clone() throws CloneNotSupportedException { Text text=null; text= (Text) super.clone(); return text; } public void setList(String str){ this.list.add(str); } public ArrayList<String> getList(){ return this.list; } }
调用:
Text text=new Text(); Text text_clone=text.clone(); text.setList("1"); text_clone.setList("2"); Log.e("qzs",text.getList()+"");
我只打印的是原生对象的集合,可是运行的结果确实:
正好印证了我前面说的,这是浅拷贝。
若是想要深拷贝,在clone()方法中改一下就能够了,其实就是对私有的变量list进行了拷贝:
@Override protected Text clone() throws CloneNotSupportedException { Text text=null; text= (Text) super.clone(); //更改的 text.list= (ArrayList<String>) this.list.clone(); return text; }
文章学习参考了《设计模式之禅》《大话设计模式》等等
另外能够加入个人Android技术交流群:458739310
你们能够关注个人微信公众号:「安卓干货铺」一个有质量、有态度的公众号!