原型模式
原型模式(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 读取二进制流。
摘自:http://www.runoob.com/design-pattern/prototype-pattern.html
浅拷贝:被拷贝对象的所有变量都含有与原对象相同的值,而且对其他对象的引用仍然是指向原来的对象,即浅拷贝只负责当前对象实例,对引用的对象不做拷贝。
深拷贝:被拷贝对象的所有的变量都含有与原来对象相同的值,除了那些引用其他对象的变量,那些引用其他对象的变量将指向一个被拷贝的新对象,而不再是原来那些被引用的对象。即深拷贝把要拷贝的对象所引用的对象也拷贝了一次。而这种对被引用到的对象拷贝叫做间接拷贝。
在决定以深拷贝的方式拷贝一个对象的时候,必须决定对间接拷贝的对象时采取浅拷贝还是深拷贝还是继续采用深拷贝。
序列化深拷贝:利用序列化来做深拷贝,把对象写到流里的过程是序列化的过程,这一过程称为“冷冻”或“腌咸菜”,反序列化对象的过程叫做“解冻”或“回鲜”。
实现
步骤 1
创建一个实现了 Prototype 接口的抽象类。
包含浅拷贝 [shallowCopy] 和深度拷贝 [deepCopy]
interface Prototype { public function shallowCopy(); public function deepCopy(); }
步骤 2
创建扩展了上面接口的实体类 ConcretePrototype。
class ConcretePrototype implements Prototype { private $_name; public function __construct($name) { $this->_name = $name; } public function setName($name) { $this->_name = $name; } public function getName() { return $this->_name; } /** * [shallowCopy 浅拷贝] */ public function shallowCopy() { return clone $this; } /** * [deepCopy 深拷贝] */ public function deepCopy() { $serialize_obj = serialize($this); $clone_obj = unserialize($serialize_obj); return $clone_obj; } }
步骤 3
创建类Demo,UsePrototype
class Demo { public $string; } class UsePrototype { public function shallow() { $demo = new Demo(); $demo->string = "susan"; $object_shallow_first = new ConcretePrototype($demo); $object_shallow_second = $object_shallow_first->shallowCopy(); var_dump($object_shallow_first->getName()); echo '<br/>'; var_dump($object_shallow_second->getName()); echo '<br/>'; $demo->string = "sacha"; var_dump($object_shallow_first->getName()); echo '<br/>'; var_dump($object_shallow_second->getName()); echo '<br/>'; } public function deep() { $demo = new Demo(); $demo->string = "Siri"; $object_deep_first = new ConcretePrototype($demo); $object_deep_second = $object_deep_first->deepCopy(); var_dump($object_deep_first->getName()); echo '<br/>'; var_dump($object_deep_second->getName()); echo '<br/>'; $demo->string = "Demo"; var_dump($object_deep_first->getName()); echo '<br/>'; var_dump($object_deep_second->getName()); echo '<br/>'; } }
步骤 4
实例化 UsePrototype 类,调用深浅拷贝方法
$up = new UsePrototype; $up->shallow(); echo '<hr>'; $up->deep();
步骤 5
执行程序,输出结果:
/* E:\websites\test\shejimoshi\Prototype Pattern\p.php:69: object(Demo)[2] public 'string' => string 'susan' (length=5) E:\websites\test\shejimoshi\Prototype Pattern\p.php:71: object(Demo)[2] public 'string' => string 'susan' (length=5) E:\websites\test\shejimoshi\Prototype Pattern\p.php:75: object(Demo)[2] public 'string' => string 'sacha' (length=5) E:\websites\test\shejimoshi\Prototype Pattern\p.php:77: object(Demo)[2] public 'string' => string 'sacha' (length=5) E:\websites\test\shejimoshi\Prototype Pattern\p.php:90: object(Demo)[4] public 'string' => string 'Siri' (length=4) E:\websites\test\shejimoshi\Prototype Pattern\p.php:92: object(Demo)[5] public 'string' => string 'Siri' (length=4) E:\websites\test\shejimoshi\Prototype Pattern\p.php:96: object(Demo)[4] public 'string' => string 'Demo' (length=4) E:\websites\test\shejimoshi\Prototype Pattern\p.php:98: object(Demo)[5] public 'string' => string 'Siri' (length=4) */