当前位置: 首页 > news >正文

重修设计模式-创建型-原型模式

重修设计模式-创建型-原型模式

原型模式就是利用已有对象(原型)通过拷贝方式来创建对象的模式,达到节省对象创建时间的目的。适用于对象创建成本较大,且同一类的不同对象之间差别不大的场景。

比如一个对象中的数据需要经过复杂计算才能得到(如排序),或者对象是从网络、文件系统等通过IO读取的,这种情况下就可以用原型模式快速拷贝出一个新对象来使用,而不是再经过复杂计算或读取 IO 来创建对象。

原型模式的核心就是对象的拷贝,且有浅拷贝和深拷贝的区别。

  • 浅拷贝:只会复制基本类型的数据和引用对象的内存地址,不会递归的拷贝引用对象本身,比如 Java 中 Object 的 clonse() 方法。
  • 深拷贝:不仅复制基本类型的数据,也会拷贝引用类型的对象(会开辟新的内存空间,并将新开辟的内存地址引用给新对象),从而得到一份完全独立的对象。

Kotlin 中 data class 的 copy() 方法,或 Java 中 Object 的 clone() 方法都是浅拷贝的实现,下面验证一下:

data class User(var name: String, var age: Int, val address: Address): Cloneable {public override fun clone(): Any {return super.clone()}
}data class Address(var street: String, var city: String) : Cloneable {override fun clone(): Any {return super.clone()}
}

测试 copy() 和 clone() 方法:

fun main(args: Array<String>) {val user = User("白泽", 18, Address("浦东新区花园石桥路28弄", "上海"))val userCopy = user.copy()val userCopy1 = user.clone() as Userprintln("user1:${user}")println("user2:${userCopy}")println("user address:${user.address == userCopy.address}")     //判断值,相当于equalprintln("user address:${user.address === userCopy.address}")    //判断内存地址println("---")println("user1:${user}")println("user2:${userCopy1}")println("user address:${user.address == userCopy1.address}")     //判断值,相当于equalprintln("user address:${user.address === userCopy1.address}")    //判断内存地址
}

代码执行结果:

user1:User(name=白泽, age=18, address=Address(street=浦东新区花园石桥路28弄, city=上海))
user2:User(name=白泽, age=18, address=Address(street=浦东新区花园石桥路28弄, city=上海))
user address:true
user address:true
---
user1:User(name=白泽, age=18, address=Address(street=浦东新区花园石桥路28弄, city=上海))
user2:User(name=白泽, age=18, address=Address(street=浦东新区花园石桥路28弄, city=上海))
user address:true
user address:true

可见引用类型对象指向的内存地址还是同一个,并没有真正地进行拷贝,这一点通过源码也可以得到验证。将 User 编译成字节码后,再反编译成 Java 语言:

public final class User implements Cloneable {@NotNullprivate String name;private int age;@NotNullprivate final Address address;//调用的是Object的clone()方法@NotNullpublic Object clone() {return super.clone();}//componentX()方法,用于解构赋值语法@NotNullpublic final User copy(@NotNull String name, int age, @NotNull Address address) {Intrinsics.checkNotNullParameter(name, "name");Intrinsics.checkNotNullParameter(address, "address");return new User(name, age, address);}
}
public final class Address implements Cloneable {@NotNullprivate String street;@NotNullprivate String city;@NotNullpublic Object clone() {return super.clone();}@NotNullpublic final Address copy(@NotNull String street, @NotNull String city) {Intrinsics.checkNotNullParameter(street, "street");Intrinsics.checkNotNullParameter(city, "city");return new Address(street, city);}
}

可以看到,copy() 方法并不会对引用类型对象进行拷贝工作,而是直接传入。而 clone() 方法则是由底层实现,执行逻辑相同。

@HotSpotIntrinsicCandidate
protected native Object clone() throws CloneNotSupportedException;

下面介绍深拷贝的几种实现方式:
1.深拷贝实现—自己实现Cloneable的clone方法:

data class User(var name: String, var age: Int, val address: Address): Cloneable {public override fun clone(): Any {return User(name, age, address.clone() as Address)}
}data class Address(var street: String, var city: String) : Cloneable {public override fun clone(): Any {return Address(street, city)}
}

打印结果:

user1:User(name=白泽, age=18, address=Address(street=浦东新区花园石桥路28弄, city=上海))
user2:User(name=白泽, age=18, address=Address(street=浦东新区花园石桥路28弄, city=上海))
user address:true
user address:false

引用地址不相同了,说明是两个独立的对象,这种方式比较麻烦,增减字段都需要对 clone() 方法进行维护,如果引用对象嵌套较深还容易出错。

2.深拷贝实现—序列化:

fun main(args: Array<String>) {val user = User("白泽", 18, Address("浦东新区花园石桥路28弄", "上海"))val userCopy = deepCopy(user) as Userprintln("user1:${user}")println("user2:${userCopy}")println("user address:${user.address == userCopy.address}")     //判断值,相当于equalprintln("user address:${user.address === userCopy.address}")    //判断内存地址
}fun deepCopy(obj: Any?): Any {val bo = ByteArrayOutputStream()val oo = ObjectOutputStream(bo)oo.writeObject(obj)val bi = ByteArrayInputStream(bo.toByteArray())val oi = ObjectInputStream(bi)return oi.readObject()
}

打印结果:

user1:User(name=白泽, age=18, address=Address(street=浦东新区花园石桥路28弄, city=上海))
user2:User(name=白泽, age=18, address=Address(street=浦东新区花园石桥路28弄, city=上海))
user address:true
user address:false

这种实现方式需要类中所有引用类型(包括嵌套的)都实现 Serializable 接口,否则会抛出 NotSerializableException 异常。

3.深拷贝实现—Json

fun main(args: Array<String>) {val user = User("白泽", 18, Address("浦东新区花园石桥路28弄", "上海"))val userCopy = deepCopy(user)println("user1:${user}")println("user2:${userCopy}")println("user address:${user.address == userCopy.address}")     //判断值,相当于equalprintln("user address:${user.address === userCopy.address}")    //判断内存地址
}inline fun <reified T> deepCopy(data: T): T {val gson = Gson()val json = gson.toJson(data)return gson.fromJson<T>(json, T::class.java)
}

打印结果:

user1:User(name=白泽, age=18, address=Address(street=浦东新区花园石桥路28, city=上海))
user2:User(name=白泽, age=18, address=Address(street=浦东新区花园石桥路28, city=上海))
user address:true
user address:false

这里使用 Kotlin 的内联函数配合范型实化封装了 deepCopy 方法,其内部是通过Google 的 Json 解析库 Gson 进行实现的,先将对象转为 Json 字符串,然后再解析 Json 来创建新对象。

4.深拷贝实现—写时复制思想

以上深拷贝方式由于在拷贝时创建了大量对象,都会伴随性能损耗。其实可以借助 Copy On Write 思想,先通过浅拷贝将对象复制出来,再使用中再进行对象的创建,从而将对象创建的损耗平摊到后续业务中。

当然这种方式需要看具体业务场景再决定如何实现,如果应用一个复杂的模式,只得到一点点的性能提升,这就是所谓的过度设计,得不偿失。

经典场景

理论说完了,再结合实际场景来尝试使用。比如 App 的个人中心展示了用户的所有信息,如用户名,生日,地址,个性签名等,同时还有保存和重置按钮,其中保存按钮需要实时更新状态:只有真正的修改了用户信息才可以点击,否则置灰无法点击。重置按钮点击后会恢复页面原信息。

这种需求场景可以利用原型模式的思想,通过拷贝得到一个新对象,后续的信息修改都基于新对象,并且每次改动后都和原对象进行对比,判断字段值是否真正修改来更新保存按钮状态。在点击重置按钮时,重新基于原对象拷贝新对象,并根据该对象刷新页面实现重置功能。

总结

利用已有对象来创建新对象,以达到节省创建时间的目的,就叫做原型模式。

原型模式的核心是对象的拷贝,对象拷贝又分为浅拷贝和深拷贝,其中浅拷贝会共用引用类型对象,而深拷贝会创建新的引用对象。

http://www.lryc.cn/news/423573.html

相关文章:

  • S71200 - 编程 - 笔记
  • 【项目】畅聊天地博客测试报告
  • 【Next】全局样式和局部样式
  • 关于Docker的详细介绍
  • 一台佳能G3811彩色喷墨打印机打印没颜色报5200的维修记录
  • 【LeetCode】452.用最少数量的箭引发气球
  • 网络安全流程规范文件解读(安全专业L1级)
  • Java、python、php版的邮件发送与过滤系统的设计与实现 (源码、调试、LW、开题、PPT)
  • st算法求RMP
  • 零基础学习Redis(1) -- Redis简介
  • 安装MySQL数据库【后端 8】
  • JAVA学习-练习试用Java实现“整数转换英文表示”
  • TPshop商城的保姆教程(Ubuntu)
  • MySQL存储过程、触发器、视图
  • 每一行txt文件的内容将作为CSV文件中的一行,逗号、空格和句号,冒号作为分隔符拆分成多列
  • 基于inotif的文件同步备份
  • luckyexcel 编辑预览excel文件
  • 记录Java使用websocket
  • (javaweb)分层解耦
  • 2024华为数通HCIP-datacom最新题库(H12-831变题更新⑨)
  • PCIe学习笔记(21)
  • 分享Embedding 模型微调的实现
  • TED: 1靶场复现【附代码】(权限提升)
  • Python(TensorFlow)衍射光学层卷积算法模拟(英伟达GPU)
  • iOS开发进阶(二十二):Xcode* 离线安装 iOS Simulator
  • Prostgresql的Timescaledb插件/扩展部署
  • 分布式知识总结(一致性Hash算法)
  • 图数据库在社交网络分析中的应用
  • Git基础使用教程
  • 技术速递|Python in Visual Studio Code 2024年8月发布