1.原型模式简介
原型模式(Prototype Pattern)是用于创建重复的对象,同时又能保证性能。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。
这种模式是实现了一个原型接口,该接口用于创建当前对象的克隆。当直接创建对象的代价比较大时,则采用这种模式。例如,一个对象需要在一个高代价的数据库操作之后被创建。我们可以缓存该对象,在下一个请求时返回它的克隆,在需要的时候更新数据库,以此来减少数据库调用。
介绍
意图:用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象。
主要解决:在运行期建立和删除原型。
何时使用: 1、当一个系统应该独立于它的产品创建,构成和表示时。 2、当要实例化的类是在运行时刻指定时,例如,通过动态装载。 3、为了避免创建一个与产品类层次平行的工厂类层次时。 4、当一个类的实例只能有几个不同状态组合中的一种时。建立相应数目的原型并克隆它们可能比每次用合适的状态手工实例化该类更方便一些。
如何解决:利用已有的一个原型对象,快速地生成和原型对象一样的实例。
关键代码: 1、实现克隆操作,在 JAVA 实现 Cloneable 接口,重写 clone(),在 .NET 中可以使用 Object 类的 MemberwiseClone() 方法来实现对象的浅拷贝或通过序列化的方式来实现深拷贝。 2、原型模式同样用于隔离类对象的使用者和具体类型(易变类)之间的耦合关系,它同样要求这些"易变类"拥有稳定的接口。
应用实例: 1、细胞分裂。 2、JAVA 中的 Object clone() 方法。
优点: 1、性能提高。 2、逃避构造函数的约束。
缺点: 1、配备克隆方法需要对类的功能进行通盘考虑,这对于全新的类不是很难,但对于已有的类不一定很容易,特别当一个类引用不支持串行化的间接对象,或者引用含有循环结构的时候。 2、必须实现 Cloneable 接口。
使用场景: 1、资源优化场景。 2、类初始化需要消化非常多的资源,这个资源包括数据、硬件资源等。 3、性能和安全要求的场景。 4、通过 new 产生一个对象需要非常繁琐的数据准备或访问权限,则可以使用原型模式。 5、一个对象多个修改者的场景。 6、一个对象需要提供给其他对象访问,而且各个调用者可能都需要修改其值时,可以考虑使用原型模式拷贝多个对象供调用者使用。 7、在实际项目中,原型模式很少单独出现,一般是和工厂方法模式一起出现,通过 clone 的方法创建一个对象,然后由工厂方法提供给调用者。原型模式已经与 Java 融为浑然一体,大家可以随手拿来使用。
注意事项:与通过对一个类进行实例化来构造新对象不同的是,原型模式是通过拷贝一个现有对象生成新对象的。浅拷贝实现 Cloneable,重写,深拷贝是通过实现 Serializable 读取二进制流。
2.原型模式的理解
以某个东西为原型复制一份(copy)也成为克隆原型;
2.1浅克隆
我们想要实现需要做到二步:
1.实现一个接口 Clone 克隆的方法
2.重写一个方法
为他重写clone 的克隆方法,其实还是依赖父类的clone 的方法;我们可以看看他的父类
(创建get/set/toString 等方法)
选择clone;
实体类
package com.example.democrud.democurd.Prototype.demo01; import lombok.AllArgsConstructor; import lombok.Data; import lombok.NoArgsConstructor; import java.util.Date; /** * 1.实现一个接口 Cloneable 克隆的方法 * * 2.重写一个方法 */ @Data @AllArgsConstructor @NoArgsConstructor public class Video implements Cloneable{ private String name; private Date creatime; //2.重写一个方法 默认调用父类的clone克隆方法 @Override protected Object clone() throws CloneNotSupportedException { return super.clone(); } @Override public String toString() { return "Video{" + "name='" + name + '\'' + ", creatime=" + creatime + '}'; } }
客户端
package com.example.democrud.democurd.Prototype.demo01; import java.util.Date; /** * 客户端 */ public class Demo { public static void main(String[] args) throws CloneNotSupportedException { //创建原型对象 Date date = new Date(); Video video = new Video("好好挣钱娶媳妇", date); System.out.println("video1==>"+video); //hashcode代表对象的地址说的是对象在hash表中的位置(也可以理解为他在房子里的那个地方摆着) System.out.println("video1=hash=>:"+video.hashCode()); System.out.println("-------------1.分割线---------------------"); //我们用上面的video 进行克隆一个对象(此处会抛出异常,强转) // 克隆出来的对象和之前是一样的 Video video2 = (Video) video.clone(); System.out.println("video2==>"+video2); System.out.println("video2=hash=>:"+video2.hashCode()); System.out.println("-------------2.分割线---------------------"); //单独修改其中一个也是可以的 video2.setName("我想躺平了"); System.out.println("video2=2=>"+video2); System.out.println("-------------3.分割线---------------------"); //我们其实只是修改了date的时候他的2个都会变化说明他们2给都是指向了同一个data的对象 date.setTime(222222222); System.out.println("video1==>"+video); System.out.println("video2==>"+video2); } }
执行结果是:(上面代码都有解释不理解请留言)
video1==>Video{name='好好挣钱娶媳妇', creatime=Thu Sep 08 00:31:07 CST 2022} video1=hash=>:-75119657 -------------1.分割线--------------------- video2==>Video{name='好好挣钱娶媳妇', creatime=Thu Sep 08 00:31:07 CST 2022} video2=hash=>:-75119657 -------------2.分割线--------------------- video2=2=>Video{name='我想躺平了', creatime=Thu Sep 08 00:31:07 CST 2022} -------------3.分割线--------------------- video1==>Video{name='好好挣钱娶媳妇', creatime=Sat Jan 03 21:43:42 CST 1970} video2==>Video{name='我想躺平了', creatime=Sat Jan 03 21:43:42 CST 1970}
2.2 深克隆
我们想实现他们的深克隆
(参考大家可以参考序列化或者非序列化也可以实现深拷贝)
上代码
package com.example.democrud.democurd.Prototype.demo01; import java.util.Date; /** * 客户端 */ public class Demo { public static void main(String[] args) throws CloneNotSupportedException { //创建原型对象 Date date = new Date(); Video video = new Video("好好挣钱娶媳妇", date); System.out.println("video1==>"+video); //hashcode代表对象的地址说的是对象在hash表中的位置(也可以理解为他在房子里的那个地方摆着) System.out.println("video1=hash=>:"+video.hashCode()); System.out.println("-------------1.分割线---------------------"); //我们用上面的video 进行克隆一个对象(此处会抛出异常,强转) // 克隆出来的对象和之前是一样的 Video video2 = (Video) video.clone(); System.out.println("video2==>"+video2); System.out.println("video2=hash=>:"+video2.hashCode()); System.out.println("-------------2.分割线---------------------"); //单独修改其中一个也是可以的 video2.setName("我想躺平了"); System.out.println("video2=2=>"+video2); System.out.println("-------------3.分割线---------------------"); //修改了date的只导致了video的值产生的变化 date.setTime(222222222); System.out.println("video1==>"+video); System.out.println("video2==>"+video2); } }
这个和上面的无区别
package com.example.democrud.democurd.Prototype.demo01; import lombok.AllArgsConstructor; import lombok.Data; import lombok.NoArgsConstructor; import java.util.Date; /** * 1.实现一个接口 Cloneable 克隆的方法 * * 2.重写一个方法 */ @Data @AllArgsConstructor @NoArgsConstructor public class Video implements Cloneable{ private String name; private Date creatime; //2.重写一个方法 默认调用父类的clone克隆方法 @Override protected Object clone() throws CloneNotSupportedException { //重写clone的方法(实现深克隆) Object clone = super.clone(); Video clone1 = (Video) clone; //对 象的属性也进行克隆 clone1.creatime= (Date) this.creatime.clone(); //此处返回clone 是因为clone1和clone的2给对象指向一个地方 return clone; } @Override public String toString() { return "Video{" + "name='" + name + '\'' + ", creatime=" + creatime + '}'; } }
运行结果:
video1==>Video{name='好好挣钱娶媳妇', creatime=Thu Sep 08 00:44:47 CST 2022} video1=hash=>:-74299985 -------------1.分割线--------------------- video2==>Video{name='好好挣钱娶媳妇', creatime=Thu Sep 08 00:44:47 CST 2022} video2=hash=>:-74299985 -------------2.分割线--------------------- video2=2=>Video{name='我想躺平了', creatime=Thu Sep 08 00:44:47 CST 2022} -------------3.分割线--------------------- video1==>Video{name='好好挣钱娶媳妇', creatime=Sat Jan 03 21:43:42 CST 1970} video2==>Video{name='我想躺平了', creatime=Thu Sep 08 00:44:47 CST 2022}
修改了data的值只是其中一个参数发生了变化他们指向的地址并不相同;我单独修改并不会影响其他的;
特别注意:
当我们想对他的对象进行克隆的时候切记这边都要进行对象的克隆;如果参数较多写起来会比较麻烦;
设计模式总目录:https://blog.csdn.net/qq_42055933/article/details/126613801?spm=1001.2014.3001.5501(查看其他章节请点击)