设计模式--原型模式

来源:互联网 发布:sqlserver入门到精通 编辑:程序博客网 时间:2024/05/29 09:49

原型模式:

       通过new产生一个对象需要非常繁琐的数据准备或访问权限,则可以使用原型模式。原型模式就是java中的克隆技术,以某个对象为原型,复制出新的对象。显然,新的对象具备原型对象的特点。

     原型模式优势:效率高(直接克隆,避免了重新执行构造过程步骤)。

     克隆类似于new,但是不同于new。new创建新的对象属性采用的是默认值。克隆出的对象的属性完全和原型对象相同。并且克隆出的新对象改变不会影响原型对象。然后,再修改克隆对象的值。

原型模式实现:

     Cloneable接口和clone方法。(Prototype模式实现起来最困难的地方就是内存复制操作,所幸在java中提供了clone方法替代我们做了绝大部分事情)

/** * 猴子 */public class Monkey implements Cloneable{private String name;private Date birthday;@Overrideprotected Object clone() throws CloneNotSupportedException {Object obj = super.clone();//浅复制,直接调用object对象的clone方法,没有调用Date对象的clone方法return obj;}public String getName() {return name;}public void setName(String name) {this.name = name;}public Date getBirthday() {return birthday;}public void setBirthday(Date birthday) {this.birthday = birthday;}public Monkey(String name, Date birthday) {super();this.name = name;this.birthday = birthday;}public Monkey() {}}/** *测试原型模式(浅复制) */public class Client {public static void main(String[] args) throws CloneNotSupportedException{Date date = new Date(500000L);Monkey dasheng = new Monkey("孙悟空",date);Monkey monkey = (Monkey)dasheng.clone();//克隆出一个新猴子System.out.println("1.名字:"+dasheng.getName()+"\t\t出生日期:"+dasheng.getBirthday());System.out.println("------------------修改出生日期--------");date.setTime(1234567890L);System.out.println("2.名字:"+dasheng.getName()+"\t\t出生日期:"+dasheng.getBirthday());monkey.setName("孙影---孙悟空的影子");System.out.println("3.名字:"+monkey.getName()+"\t\t出生日期:"+monkey.getBirthday());}}
     运行的结果如下:


      由上面运行结果,可知,当修改出生日期date的值,孙悟空和孙影的出生日期的值都跟着改变了,这说明dasheng和monkey对象共同指向了同一个date对象,这就属于浅复制。

      深复制就是当克隆时,把属性对象都重新分配属性对象,也就说克隆前对象和克隆后对象的属性分别指向了不同的地址空间。深复制实现如下:

 

/** * 猴子 */public class Monkey implements Cloneable{private String name;private Date birthday;@Overrideprotected Object clone() throws CloneNotSupportedException {Object obj = super.clone();//直接调用object对象的clone方法//添加如下代码实现深复制Monkey monkey = (Monkey)obj;monkey.birthday = (Date)this.birthday.clone();//把Date属性也进行复制return monkey;}public String getName() {return name;}public void setName(String name) {this.name = name;}public Date getBirthday() {return birthday;}public void setBirthday(Date birthday) {this.birthday = birthday;}public Monkey(String name, Date birthday) {super();this.name = name;this.birthday = birthday;}public Monkey() {}}/** *测试原型模式(深复制) */public class Client {public static void main(String[] args) throws CloneNotSupportedException{Date date = new Date(500000L);Monkey dasheng = new Monkey("孙悟空",date);Monkey monkey = (Monkey)dasheng.clone();//克隆出一个新猴子System.out.println("1.名字:"+dasheng.getName()+"\t\t出生日期:"+dasheng.getBirthday());System.out.println("------------------修改出生日期--------");date.setTime(1234567890L);System.out.println("2.名字:"+dasheng.getName()+"\t\t出生日期:"+dasheng.getBirthday());monkey.setName("孙影---孙悟空的影子");System.out.println("3.名字:"+monkey.getName()+"\t\t出生日期:"+monkey.getBirthday());}}
      运行结果如下:


     原型模式的注意事项:
1.使用原型模式复制对象不会调用类的构造方法。因为对象的复制是通过调用Object类的clone方法来完成的,它直接在内存中复制数据,因此不会调用到类的构造方法。不但构造方法中的代码不会执行,甚至连访问权限都对原型模式无效。
2.深拷贝与浅拷贝。Object类的clone方法只会拷贝对象中的基本的数据类型,对于数组、容器对象、引用对象等都不会拷贝,这就是浅拷贝。如果要实现深拷贝,必须将原型模式中的数组、容器对象、引用对象等另行拷贝。

0 0
原创粉丝点击