java设计模式--原型模式详解
引例
问题:
现在有一只羊(包含属性:名字Dolly、年龄2),需要克隆10只属性完全相同的羊。
一般解法:
定义Sheep类表示羊,包括构造器、getter()和toString()。
public class Sheep { private String name; private int age; public Sheep(String name, int age) { this.name = name; this.age = age; } public String getName() { return name; } public int getAge() { return age; } @Override public String toString() { return "Sheep{" + "name='" + name + '\'' + ", age=" + age + '}'; } }
在客户端实例化多利,然后再根据多利的属性去实例化10只羊。
public class Client { public static void main(String[] args) { Sheep sheepDolly=new Sheep("Dolly",2); Sheep sheep1 = new Sheep(sheepDolly.getName(), sheepDolly.getAge()); Sheep sheep2 = new Sheep(sheepDolly.getName(), sheepDolly.getAge()); Sheep sheep3 = new Sheep(sheepDolly.getName(), sheepDolly.getAge()); //.... System.out.println(sheep1+",hashCode:"+sheep1.hashCode()); System.out.println(sheep2+",hashCode:"+sheep2.hashCode()); System.out.println(sheep3+",hashCode:"+sheep3.hashCode()); //... } }
运行结果
优缺点:
这种方法是我们首先很容易就能想到的,也是绝大多数人的第一做法。
但缺点也很明显,每次创建新对象时需要获取原始对象的属性,对象复杂时效率很低;此外不能动态获得对象运行时的状态,若类增减属性需要改动代码。
下面我们看下原型模式的解法。
原型模式
原型模式(Prototype Pattern)是一种创建型设计模式,允许一个对象再创建另外一个可定制的对象,无需知道如何创建的细节。即用原型实例指定创建对象的种类,并且通过拷贝这些原型,创建新的对象。
工作原理:将原型对象传给那个要发动创建的对象,这个要发动创建的对象通过请求原型对象拷贝它们自己来实施创建。即用基类Object的clone()方法或序列化。
UML类图:
- Prototype:原型类,声明一个克隆自己的接口
- ConcretePrototype: 具体的原型类, 实现一个克隆自己的操作
- Client: 客户端让一个原型对象克隆自己,从而创建一个新的对象
原型模式又可分为浅拷贝和深拷贝,区别在于对引用数据类型的成员变量的拷贝,小朋友你是否有很多问号? 不急 ,看完这两种方法实现你就懂了。
浅拷贝
在原先Sheep类基础上实现Cloneable接口,重写clone方法。
public class Sheep implements Cloneable{ private String name; private int age; @Override protected Object clone() {//克隆该实例,使用默认的clone方法来完成 Sheep sheep = null; try { sheep = (Sheep)super.clone(); } catch (Exception e) { System.out.println(e.getMessage()); } return sheep; } public Sheep(String name, int age) { this.name = name; this.age = age; } @Override public String toString() { return "Sheep{" + "name='" + name + '\'' + ", age=" + age + '}'; } }
客户端调用
public class Client { public static void main(String[] args) { Sheep sheepDolly=new Sheep("Dolly",2); Sheep sheep1 = (Sheep)sheepDolly.clone(); Sheep sheep2 = (Sheep)sheepDolly.clone(); Sheep sheep3 = (Sheep)sheepDolly.clone(); //.... System.out.println("sheep1:"+sheep1+",hashCode:" + sheep1.hashCode()); System.out.println("sheep2:"+sheep2+",hashCode:" + sheep2.hashCode()); System.out.println("sheep3:"+sheep3+",hashCode:" + sheep3.hashCode()); //... } }
运行结果
至此,原型模式的浅拷贝也成功克隆了三个对象,但是看进度条发现并不简单。
现在小羊有了一个朋友小牛,Sheep类添加了一个引用属性Cow,我们同样再克隆一遍。
Sheep类
public class Sheep implements Cloneable{ private String name; private int age; public Cow friend;//新朋友Cow对象,其余不变 @Override protected Object clone() { Sheep sheep = null; try { sheep = (Sheep)super.clone(); } catch (Exception e) { System.out.println(e.getMessage()); } return sheep; } public Sheep(String name, int age) { this.name = name; this.age = age; } @Override public String toString() { return "Sheep{" + "name='" + name + '\'' + ", age=" + age + '}'; } }
新添的Cow类
public class Cow { private String name; private int age; public Cow(String name, int age) { this.name = name; this.age = age; } @Override public String toString() { return "Cow{" + "name='" + name + '\'' + ", age=" + age + '}'; } }
客户端调用克隆
public class Client { public static void main(String[] args) { Sheep sheepDolly=new Sheep("Dolly",2); sheepDolly.friend=new Cow("Tom",1); //并实例化朋友 Sheep sheep1 = (Sheep)sheepDolly.clone(); Sheep sheep2 = (Sheep)sheepDolly.clone(); Sheep sheep3 = (Sheep)sheepDolly.clone(); //.... System.out.println("sheep1:"+sheep1+",hashCode:" + sheep1.hashCode()); System.out.println("sheep1.friend:"+sheep1.friend+",hashCode:" + sheep1.friend.hashCode()+'\n'); System.out.println("sheep2:"+sheep2+",hashCode:" + sheep2.hashCode()); System.out.println("sheep2.friend:"+sheep2.friend+",hashCode:" + sheep2.friend.hashCode()+'\n'); System.out.println("sheep3:"+sheep3+",hashCode:" + sheep3.hashCode()); System.out.println("sheep3.friend:"+sheep3.friend+",hashCode:" + sheep3.friend.hashCode()+'\n'); //... } }
运行结果
通过运行结果发现,浅拷贝通过Object的clone()成功克隆实例化了三个新对象,但是并没有克隆实例化对象中的引用属性,也就是没有克隆friend对象(禁止套娃 ),三个新克隆对象的friend还是指向原克隆前的friend,即同一个对象。
这样的话,他们四个的friend是引用同一个,若一个对象修改了friend属性,势必会影响其他三个对象的该成员变量值。
小结:
- 浅拷贝是使用默认的 clone()方法来实现
- 基本数据类型的成员变量,浅拷贝会直接进行值传递(复制属性值给新对象)。
- 引用数据类型的成员变量,浅拷贝会进行引用传递(复制引用值(内存地址)给新对象)。
深拷贝
方法一:
机灵的人儿看出,再clone一遍cow不就好了,但是手动递归下去不推荐。
1.Cow类也实现Cloneable接口
public class Cow implements Cloneable{ private String name; private int age; public Cow(String name, int age) { this.name = name; this.age = age; } //无引用类型,直接clone即可 @Override protected Object clone() throws CloneNotSupportedException { return super.clone(); //直接抛出了,没用try-catch } @Override public String toString() { return "Cow{" + "name='" + name + '\'' + ", age=" + age + '}'; } }
Sheep类的clone再添加调用cow的clone
public class Sheep implements Cloneable{ private String name; private int age; public Cow friend;//新朋友Cow对象,其余不变 @Override protected Object clone() throws CloneNotSupportedException { Object deep = null; //完成对基本数据类型(属性)和String的克隆 deep = super.clone(); //对引用类型的属性,进行再次clone Sheep sheep = (Sheep)deep; sheep.friend = (Cow)friend.clone(); return sheep; } public Sheep(String name, int age) { this.name = name; this.age = age; } @Override public String toString() { return "Sheep{" + "name='" + name + '\'' + ", age=" + age + '}'; } }
客户端调用
public class Client { public static void main(String[] args) throws CloneNotSupportedException { Sheep sheepDolly=new Sheep("Dolly",2); sheepDolly.friend=new Cow("Tom",1); //并实例化朋友 Sheep sheep1 = (Sheep)sheepDolly.clone(); Sheep sheep2 = (Sheep)sheepDolly.clone(); Sheep sheep3 = (Sheep)sheepDolly.clone(); //.... System.out.println("sheep1:"+sheep1+",hashCode:" + sheep1.hashCode()); System.out.println("sheep1.friend:"+sheep1.friend+",hashCode:" + sheep1.friend.hashCode()+'\n'); System.out.println("sheep2:"+sheep2+",hashCode:" + sheep2.hashCode()); System.out.println("sheep2.friend:"+sheep2.friend+",hashCode:" + sheep2.friend.hashCode()+'\n'); System.out.println("sheep3:"+sheep3+",hashCode:" + sheep3.hashCode()); System.out.println("sheep3.friend:"+sheep3.friend+",hashCode:" + sheep3.friend.hashCode()+'\n'); //... } }
运行结果
方法二:
通过对象序列化实现深拷贝(推荐)
1.Cow类实现序列化接口,不必实现Cloneable接口了
public class Cow implements Serializable { private String name; private int age; public Cow(String name, int age) { this.name = name; this.age = age; } @Override public String toString() { return "Cow{" + "name='" + name + '\'' + ", age=" + age + '}'; } }
2.在Sheep类实现序列化接口
public class Sheep implements Serializable { //实现序列化接口 private String name; private int age; public Cow friend; public Sheep(String name, int age) { this.name = name; this.age = age; } @Override public String toString() { return "Sheep{" + "name='" + name + '\'' + ", age=" + age + '}'; } public Object deepClone() { //深拷贝 //创建流对象 ByteArrayOutputStream bos = null; ObjectOutputStream oos = null; ByteArrayInputStream bis = null; ObjectInputStream ois = null; try { //序列化 bos = new ByteArrayOutputStream(); oos = new ObjectOutputStream(bos); oos.writeObject(this); //当前这个对象以对象流的方式输出 //反序列化 bis = new ByteArrayInputStream(bos.toByteArray()); ois = new ObjectInputStream(bis); Sheep sheep = (Sheep) ois.readObject(); return sheep; } catch (Exception e) { e.printStackTrace(); return null; } finally { //关闭流 try { bos.close(); oos.close(); bis.close(); ois.close(); } catch (Exception e2) { System.out.println(e2.getMessage()); } } } }
3.客户端调用
public class Client { public static void main(String[] args) throws CloneNotSupportedException { Sheep sheepDolly=new Sheep("Dolly",2); sheepDolly.friend=new Cow("Tom",1); //并实例化朋友 Sheep sheep1 = (Sheep)sheepDolly.deepClone(); Sheep sheep2 = (Sheep)sheepDolly.deepClone(); Sheep sheep3 = (Sheep)sheepDolly.deepClone(); //.... System.out.println("sheep1:"+sheep1+",hashCode:" + sheep1.hashCode()); System.out.println("sheep1.friend:"+sheep1.friend+",hashCode:" + sheep1.friend.hashCode()+'\n'); System.out.println("sheep2:"+sheep2+",hashCode:" + sheep2.hashCode()); System.out.println("sheep2.friend:"+sheep2.friend+",hashCode:" + sheep2.friend.hashCode()+'\n'); System.out.println("sheep3:"+sheep3+",hashCode:" + sheep3.hashCode()); System.out.println("sheep3.friend:"+sheep3.friend+",hashCode:" + sheep3.friend.hashCode()+'\n'); //... } }
运行结果
原型模式总结:
- 创建新的对象比较复杂时,可以利用原型模式简化对象的创建过程,同时也能够提高效率
- 可以不用重新初始化对象,动态地获得对象运行时的状态。
- 如果原始对象发生变化(增加或者减少属性),其它克隆对象的也会发生相应的变化,无需修改代码
- 若成员变量无引用类型,浅拷贝clone即可;若引用类型的成员变量很少,可考虑递归实现clone,否则推荐序列化。
总结
本篇文章就到这里了,希望能给你带来帮助,也希望您能够多多关注猪先飞的更多内容!
相关文章
- 这篇文章主要介绍了如何利用java语言实现经典《复杂迷宫》游戏,文中采用了swing技术进行了界面化处理,感兴趣的小伙伴可以动手试一试...2022-02-01
java 运行报错has been compiled by a more recent version of the Java Runtime
java 运行报错has been compiled by a more recent version of the Java Runtime (class file version 54.0)...2021-04-01- 这篇文章主要介绍了在java中获取List集合中最大的日期时间操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2020-08-15
- 这篇文章主要介绍了教你怎么用Java获取国家法定节假日,文中有非常详细的代码示例,对正在学习java的小伙伴们有非常好的帮助,需要的朋友可以参考下...2021-04-23
- 这篇文章主要介绍了Java如何发起http请求的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2021-03-31
- 说起C#和Java这两门语言(语法,数据类型 等),个人以为,大概有90%以上的相似,甚至可以认为几乎一样。但是在工作中,我也发现了一些细微的差别...2020-06-25
- 神马是“解释器模式”?先翻开《GOF》看看Definition:给定一个语言,定义它的文法的一种表示,并定义一个解释器,这个解释器使用该表示来解释语言中的句子。在开篇之前还是要科普几个概念: 抽象语法树: 解释器模式并未解释如...2014-06-07
- 这篇文章主要介绍了解决Java处理HTTP请求超时的问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2021-03-29
- 这篇文章主要介绍了java 判断两个时间段是否重叠的案例,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2020-08-15
- 这篇文章主要介绍了超简洁java实现双色球若干注随机号码生成(实例代码),本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下...2021-04-02
java 画pdf用itext调整表格宽度、自定义各个列宽的方法
这篇文章主要介绍了java 画pdf用itext调整表格宽度、自定义各个列宽的方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2021-01-31- 这篇文章主要介绍了Java生成随机姓名、性别和年龄的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2020-10-01
- 这篇文章主要为大家介绍了JavaScript设计模式中的装饰者模式,对JavaScript设计模式感兴趣的小伙伴们可以参考一下...2016-01-21
- 这篇文章主要介绍了java正则表达式判断前端参数修改表中另一个字段的值,需要的朋友可以参考下...2021-05-07
Java使用ScriptEngine动态执行代码(附Java几种动态执行代码比较)
这篇文章主要介绍了Java使用ScriptEngine动态执行代码,并且分享Java几种动态执行代码比较,需要的朋友可以参考下...2021-04-15- 这篇文章主要介绍了Java开发实现人机猜拳游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...2020-08-03
- 这篇文章主要介绍了Java List集合返回值去掉中括号('[ ]')的操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2020-08-29
Java中lombok的@Builder注解的解析与简单使用详解
这篇文章主要介绍了Java中lombok的@Builder注解的解析与简单使用,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下...2021-01-06- 下面小编就为大家带来一篇java中String类型变量的赋值问题介绍。小编觉得挺不错的。现在分享给大家,给大家一个参考。...2016-03-28
Java 8 Stream 的终极技巧——Collectors 功能与操作方法详解
这篇文章主要介绍了Java 8 Stream Collectors 功能与操作方法,结合实例形式详细分析了Java 8 Stream Collectors 功能、操作方法及相关注意事项,需要的朋友可以参考下...2020-05-20