一、什么是克隆模式
故事剧情——克隆模式
我想克隆一个自己,可以一边敲代码,一边看书,一边聊天......
用程序模拟生活
from copy import copy, deepcopy
class Person:
"""人"""
def __init__(self, name, age):
self.__name = name
self.__age = age
def showMyself(self):
print("我是" + self.__name + ", 年龄" + str(self.__age) + "。")
def coding(self):
print("我是码农,我用程序改变世界,coding......")
def reading(self):
print("阅读使我快乐!,知识使我成长!如饥似渴地阅读是生活的一部分......")
def fallInLove(self):
print("春风吹,月亮明,花前月下好相约......")
def clone(self):
return copy(self)
def testClone():
tony = Person("Tony", 27)
tony.showMyself()
tony.coding()
tony1 = tony.clone()
tony1.showMyself()
tony1.reading()
tony2 = tony.clone()
tony2.showMyself()
tony2.fallInLove()
if __name__ == "__main__":
testClone()
输出结果:
我是Tony, 年龄27。
我是码农,我用程序改变世界,coding......
我是Tony, 年龄27。
阅读使我快乐!,知识使我成长!如饥似渴地阅读是生活的一部分......
我是Tony, 年龄27。
春风吹,月亮明,花前月下好相约......
在上面的代码中,Tony克隆出两个自己tony1和tony2,因为是克隆出来的,所有的姓名和年龄都一样,这样Tony就可以同时敲代码、读书和约会。
用go来演示:
package main
import (
"fmt"
"strconv"
)
type Person struct {
name string
age int
}
func (p *Person) showMyself() {
fmt.Println("我是" + p.name + ",年龄" + strconv.Itoa(p.age) + "。")
}
func (p *Person) coding() {
fmt.Println("我是码农,我用程序改变世界,coding......")
}
func (p *Person) reading() {
fmt.Println("阅读是我快乐!知识使我成长!如饥似渴地阅读是生活的一部分......")
}
func (p *Person) fallInLove() {
fmt.Println("春风吹,月亮明,花前月下好相约......")
}
func (p *Person) clone() *Person {
q := *p
return &q
}
func main() {
tony := &Person{
name: "Tony",
age: 27,
}
tony.showMyself()
tony.coding()
tony1 := tony.clone()
tony1.showMyself()
tony.reading()
tony2 := tony.clone()
tony2.showMyself()
tony2.fallInLove()
}
二、什么是克隆模式
2.1 克隆模式的定义
用原型实例指定要创建对象的种类,并通过拷贝这些原型属性来创建新的对象。通过拷贝自身的属性来创建一个新对象的过程叫做作克隆模式。
很多书籍中克隆模式也被称为原型模式。克隆模式的核心就是一个clone方法,clone方法的功能就是拷贝父本的所有属性。主要包括两个过程:
(1)分配一块新的内存空间给新的对象。
(2)拷贝父本对象的所有属性。
2.2 浅拷贝与深拷贝
浅拷贝只拷贝引用类型对象的指针(指向),而不拷贝引用类型对象指向的值;深拷贝则同时拷贝引用类型对象及其指向的值。
而深拷贝和浅拷贝也可以这样理解:
深拷贝就是拷贝整个对象,源对象和拷贝对象没有任何关联,也不会受到任何影响
浅拷贝就是拷贝对象指针,其实是引用地址都一样,所以属于牵一发动全身
三、克隆模式的模型抽象
from copy import deepcopy, copy
class Clone:
"""克隆的基类"""
def clone(self):
"""浅拷贝的方式克隆对象"""
return copy(self)
def deepClone(self):
"""深拷贝的方式克隆对象"""
return deepcopy(self)
copy代表的是浅拷贝,deepcopy是深拷贝。
package main
import "encoding/json"
type Clone struct {
name string
hobby []string
}
func (c *Clone) clone() *Clone {
d := *c
return &d
}
func (c *Clone) deepClone() (*Clone, error) {
var dst = new(Clone)
b, err := json.Marshal(c)
if err != nil {
return nil, err
}
err = json.Unmarshal(b, dst)
return dst, err
}
//func deepCopy(dst, src interface{}) error {
// var buf bytes.Buffer
// if err := gob.NewEncoder(&buf).Encode(src); err != nil {
// return err
// }
// return gob.NewDecoder(bytes.NewBuffer(buf.Bytes())).Decode(dst)
//}
Go语言中所有赋值操作都是值传递,如果结构中不含指针,则直接赋值就是深度拷贝;如果结构中含有指针(包括自定义指针,以及切片,map等使用了指针的内置类型),则数据源和拷贝之间对应指针会共同指向同一块内存,这时深度拷贝需要特别处理。目前,有三种方法,一是用gob序列化成字节序列再反序列化生成克隆对象;二是先转换成json字节序列,再解析字节序列生成克隆对象;三是针对具体情况,定制化拷贝。前两种方法虽然比较通用但是因为使用了reflex反射,性能比定制化拷贝要低出2个数量级,所以在性能要求较高的情况下应该尽量避免使用前两者。
四、模型说明
4.1 设计要点
克隆模式也叫原型模式。在设计克隆模式时,唯一需要注意的是:区分深拷贝与浅拷贝,除非一些特殊情况(如需求本身就要求两个对象一起改变),尽量使用深拷贝的方式。
4.2 克隆模式的优缺点
(1)克隆模式通过内存拷贝的方式进行复制,比new的方式创建对象性能更好。
(2)通过深拷贝的方式,可以方便地创建一个具有相同属性和行为的另一个对象,特别是对于复杂对象,方便性尤为突出。
缺点:
通过克隆的方式创建对象,不会执行类的初始化函数(init)(这一点是说在pyton中,go 语言没有init函数)。这不一定是缺点,但大家使用的时候需要注意这一点。
4.3 应用场景
(1)如果创建新对象(如复杂对象)成本较高,我们可以利用已有的对象进行复制来获得。
(2)类的初始化需要消耗非常多的资源时,如需要消耗很多的数据、硬件等资源。
(3)可配合备忘录模式做一些备份的工作。
摘录来自
人人都懂设计模式:从生活中领悟设计模式:Python实现