原型模式
概述
用一个已经创建的实例作为原型,通过复制该原型对象来创建一个和原型对象相同的新对象
结构
原型模式包含如下角色:
- 抽象原型类:规定了具体原型对象必须实现的的NSCopying()方法。
- 具体原型类:实现抽象原型类的 NSCopying方法,它是可被复制的对象。
- 访问类:使用具体原型类中的 NSCopying方法来复制新的对象。
实现
原型模式的克隆分为浅克隆和深克隆。
浅克隆:创建一个新对象,新对象的属性和原来对象完全相同,对于非基本数据类型属性,仍指向原有属性所指向的对象的内存地址。
深克隆:创建一个新对象,属性中引用的其他对象也会被克隆,不再指向原有对象地址。
class RealizeType: NSCopying{
private var intValue = 1
private var stringValue = "Value"
required init() {
print("具体的原型对象创建完成了")
}
func copy(with zone: NSZone? = nil) -> Any {
let prototype = type(of: self).init()
prototype.intValue = self.intValue
prototype.stringValue = self.stringValue
print("具体原型复制成功")
return prototype
}
}
测试代码
var realizeType = RealizeType()
var realizeType1 = realizeType.copy(with: nil)
运行结果:
可以看出 两个对象的内存地址完全不同拷贝出了一个新的对象。
案例
同一学校的三好学生奖状除了获奖人姓名不同,其他都相同,可以使用原型模式复制多个“三好学生”奖状出来,然后修改奖状上的名字即可。
//三好学生姓名
class Student :NSCopying{
required init(){
}
var name:String = ""
func copy(with zone: NSZone? = nil) -> Any {
let prototype = type(of: self).init()
prototype.name = name
print("具体原型复制成功")
return prototype
}
func toString()->String{
return "student{\n" + "name = " + name + "\n}"
}
}
class Citation:NSCopying{
var student:Student
required init(student:Student){
self.student = student
}
func copy(with zone: NSZone? = nil) -> Any {
let prototype = type(of: self).init(student:Student())
prototype.student = student.copy(with: nil) as! Student
print("具体原型复制成功")
return prototype
}
func show(){
print(student.name + "同学,在2022年第一学期中表现优秀,被评为三好学生。特发此状,以示鼓励!")
}
}
测试代码
//创建原型对象
var citation = Citation(student: Student())
citation.student.name = "张三"
var citation1:Citation = citation.copy(with: nil) as! Citation
citation1.student.name = "李四"
//citation.name = "张三"
//citation.name = "李四"
print(1)
运行结果
此处需注意引用对象的深拷贝和浅拷贝问题
原型模式使用场景
- 对象的创建非常复杂的话,可以使用原型模式快捷的创建对象
- 性能和安全性要求高
感谢黑马程序员课程