原型模式

基本理解

  • 原型模式(Prototype),用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象。
  • 原型模式其实就是从一个对象再创建另外一个可定制的对象,而且不需要知道任何创建的细节。
  • 一般在初始化信息不发生变化的情况下,克隆是最好的办法,这既隐藏了对象创建的细节,又对性能是大大的提高。
  • 不用重新初始化对象,而是动态地获得了对象运行时的状态。
  • 深复制和浅复制
    • 浅复制:被复制对象的所有变量都含有与原来对象相同的值,而所有的对其他对象的引用都仍然指向原来对象。
    • 深复制:把引用对象的变量指向复制过的新对象,而不是原来的被引用的对象。

何时使用原型模式

  • 需要创建的对象应独立于其类型与创建方式。
  • 要实例化的类是在运行时决定的。
  • 不想要与产品层次相对应的工厂层次。
  • 不同类的实例间的差别仅是状态的若干组合。因此复制相应数量的原型比手工实例话更加方便。
  • 类不容易创建,比如每个组件可把其他组件作为子节点的组合对象。复制已有的组合对象并对副本进行修改会更容易。

说明

  • 不管是什么对象,只要复制自身比手工实例化要好,都可以是原型对象。
  • 使用此模式的情形:
    • 有很多相关的类,其行为略有不同,而且主要差异在于内部属性,如名称、图像等。
    • 需要使用组合(树形)对象作为其他东西的基础,例如使用组合对象作为组件来构建另一个组合对象。

深复制和浅复制

指针只是存储在内存中资源地址的占位符。在复制操作中,如果只是将指针复制给新对象(副本),那么底层的资源实际上仍然由两个实例在共享。其实两者指向的都是同一块内存。这就是浅复制。



什么是深复制呢?深复制是指不仅复制指针值,还复制指针所指向的资源。

使用Cocoa Touch框架中的对象复制

Cocoa Touch框架为NSObject的派生类提供了实现深复制的协议。NSObject的子类需要实现NSCopying协议以及其方法:

-(id)copyWithZone:(NSZone *)zone;

NSObject有一个实列方法叫做-(id)copy。默认的copy方法调用[self copyWithZone:nil];对于采纳了NSCopying协议的子类,需要实现这个方法,否则会引发异常。

例子

Prototype.h

//
// Prototype.h
// Test
//
// Created by zhanggui on 15/8/5.
// Copyright (c) 2015年 zhanggui. All rights reserved.
// #import <Foundation/Foundation.h> @interface Prototype : NSObject<NSCopying> @property(nonatomic)NSString *name; @end

Prototype.m

//
// Prototype.m
// Test
//
// Created by zhanggui on 15/8/5.
// Copyright (c) 2015年 zhanggui. All rights reserved.
// #import "Prototype.h" @implementation Prototype //实现NSCopying中的方法
-(id)copyWithZone:(NSZone *)zone
{
Prototype *p = [[Prototype allocWithZone:zone] init];
return p;
}
@end

然后在ViewController中调用。具体的viewDidLoad代码如下:

- (void)viewDidLoad {
[super viewDidLoad]; Prototype *prototype = [[Prototype alloc] init];
prototype.name = @"originName";
//deepCopy
Prototype *prototypeDeepCopy = [prototype copy]; //shallowCopye
Prototype *prototypSallowCopy = prototype; NSLog(@"prototyp addreess:%p",prototype);
NSLog(@"deepAddress:%p",prototypeDeepCopy);
NSLog(@"shallowAddress:%p",prototypSallowCopy); NSLog(@"prototype:%@",prototype.name);
NSLog(@"deep:%@",prototypeDeepCopy.name);
NSLog(@"shallow:%@",prototypSallowCopy.name); NSLog(@"changed.........");
prototype.name = @"newName";
NSLog(@"prototype:%@",prototype.name);
NSLog(@"deep:%@",prototypeDeepCopy.name);
NSLog(@"shallow:%@",prototypSallowCopy.name); }

运行结果如下:

2015-08-05 20:51:20.535 Test[8763:3496263] prototyp addreess:0x7a3e35a0
2015-08-05 20:51:20.536 Test[8763:3496263] deepAddress:0x7a3e4650
2015-08-05 20:51:20.536 Test[8763:3496263] shallowAddress:0x7a3e35a0
2015-08-05 20:51:20.536 Test[8763:3496263] prototype:originName
2015-08-05 20:51:20.536 Test[8763:3496263] deep:(null)
2015-08-05 20:51:20.536 Test[8763:3496263] shallow:originName
2015-08-05 20:51:20.536 Test[8763:3496263] changed.........
2015-08-05 20:51:20.536 Test[8763:3496263] prototype:newName
2015-08-05 20:51:20.536 Test[8763:3496263] deep:(null)
2015-08-05 20:51:20.536 Test[8763:3496263] shallow:newName

结论:

  • 使用copyWithZone:(NSZone *)zone方法实现了深复制,通过copy方法(该方法默认调用copyWithZone方法)复制得到prototypeDeepCopy,从结果可以看出:深复制对象和和源对象的地址是不一样的:
  • 2015-08-05 20:51:20.535 Test[8763:3496263] prototyp addreess:0x7a3e35a0
  • 2015-08-05 20:51:20.536 Test[8763:3496263] deepAddress:0x7a3e4650

而浅复制对象和源对象地址是一样的:

  • 2015-08-05 20:51:20.535 Test[8763:3496263] prototyp addreess:0x7a3e35a0

  • 2015-08-05 20:51:20.536 Test[8763:3496263] shallowAddress:0x7a3e35a0

    这说明如果是深复制就会开辟一块新的内存。对象所有的属性都是null的。

  • 对于浅复制的对象,一旦源对象的属性改变,那么浅复制的对象也会跟着改变。

附源码下载地址:

http://pan.baidu.com/s/1kTByijH

iOS设计模式之原型模式的更多相关文章

  1. iOS设计模式-原型模式

    | 导语 定义:用原型实例指定创建对象的种类,并通过拷贝这些原型创建新的对象. 通过深复制(克隆.拷贝)一个指定类型的对象来创建更多同类型的对象.这个指定的对象可被称为“原型”对象,也就是通过复制原型 ...

  2. 设计模式_11_原型模式(prototype)深拷贝、浅拷贝

    设计模式_11_原型模式(prototype) 浅拷贝: package designPatternOf23; /** * 定义:用原型实例,指定创建对象的种类,并通过拷贝这些原型创建新的对象 * P ...

  3. C#设计模式(6)——原型模式(Prototype Pattern)

    一.引言 在软件系统中,当创建一个类的实例的过程很昂贵或很复杂,并且我们需要创建多个这样类的实例时,如果我们用new操作符去创建这样的类实例,这未免会增加创建类的复杂度和耗费更多的内存空间,因为这样在 ...

  4. iOS 设计模式之工厂模式

    iOS 设计模式之工厂模式 分类: 设计模式2014-02-10 18:05 11020人阅读 评论(2) 收藏 举报 ios设计模式 工厂模式我的理解是:他就是为了创建对象的 创建对象的时候,我们一 ...

  5. 乐在其中设计模式(C#) - 原型模式(Prototype Pattern)

    原文:乐在其中设计模式(C#) - 原型模式(Prototype Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 原型模式(Prototype Pattern) 作者:weba ...

  6. C#设计模式之六原型模式(Prototype)【创建型】

    一.引言 在开始今天的文章之前先说明一点,欢迎大家来指正.很多人说原型设计模式会节省机器内存,他们说是拷贝出来的对象,这些对象其实都是原型的复制,不会使用内存.我认为这是不对的,因为拷贝出来的每一个对 ...

  7. C#设计模式之五原型模式(Prototype Pattern)【创建型】

    一.引言 在开始今天的文章之前先说明一点,欢迎大家来指正.很多人说原型设计模式会节省机器内存,他们说是拷贝出来的对象,这些对象其实都是原型的复制,不会使用内存.我认为这是不对的,因为拷贝出来的每一个对 ...

  8. C#设计模式(6)——原型模式(Prototype Pattern)(转)

    一.引言 在软件系统中,当创建一个类的实例的过程很昂贵或很复杂,并且我们需要创建多个这样类的实例时,如果我们用new操作符去创建这样的类实例,这未免会增加创建类的复杂度和耗费更多的内存空间,因为这样在 ...

  9. C#设计模式(6)——原型模式(Prototype Pattern) C# 深浅复制 MemberwiseClone

    C#设计模式(6)——原型模式(Prototype Pattern)   一.引言 在软件系统中,当创建一个类的实例的过程很昂贵或很复杂,并且我们需要创建多个这样类的实例时,如果我们用new操作符去创 ...

随机推荐

  1. STM32 flash 内存分布介绍

    摘要: 本文以STM32F103RBT6为例介绍了片上Flash(Embedded Flash)若干问题,包括Flash大小(内存映射).块大小.页面大小.寄存器.这些知识,有利于写Flash驱动. ...

  2. Reveal查看任意app的高级技巧(转)

    原文:http://zhuanlan.zhihu.com/iOSRe/19646016 Reveal查看任意app的高级技巧 hangcom · 12 小时前 Reveal是一个很强大的UI分析工具, ...

  3. IT痴汉的工作现状24-Just for fun

    早在大学一开始我进行Linux的学习了,那时大家都跟Windows Xp玩的火热,而我从来就不走寻常路,在XP上安装了VMware虚拟机搞起了Linux的探索.这简直让我眼界大开,每天都和那么多的国外 ...

  4. MVC显示Base64图片

    本篇演示ASP.NET MVC应用程序,显示Base64图片. Insus.NET浏览网页,发现一个站点http://www.base64-image.de/ 想起以前也有实现过<如何把数据流转 ...

  5. MPU9250调试

    MPU9250 芯片概述 MPU9250芯片是一个9轴姿态传感芯片,其中包含了3轴加速度传感器.3轴角速度传感器以及三轴磁力计. 其本质上是MPU6050芯片+AK8963. 可以获取传感芯片的加速度 ...

  6. SharePoint 2013 搜索爬网功能

    最近在政府部门介绍SharePoint 2013 新功能,我也准备了很多,比如SharePoint 2013的Search.以后有机会谈谈Office Web App,Workflow等. Share ...

  7. Fenix – 基于 Node.js 的桌面静态 Web 服务器

    Fenix 是一个提供给开发人员使用的简单的桌面静态 Web 服务器,基于 Node.js 开发.您可以同时在上面运行任意数量的项目,特别适合前端开发人员使用. 您可以通过免费的 Node.js 控制 ...

  8. Koa – 更加强大的下一代 Node.js Web 框架

    Koa 是 Express 的开发团队设计的下一代 Web 框架,其目的是为 Web 应用程序提供更小,更具表现力,更坚实的基础.Koa 没有核捆绑任何中间件,并提供了一​​套优雅的方法,使服务器端开 ...

  9. E:无法修正错误,因为您要求某些软件包保持现状,就是它们破坏了软件包间的依赖关系

    安装terminator等一些软件等时候,遇到了这样等问题 leo@leo:~$ sudo apt-get install terminator [sudo] password for leo: 正在 ...

  10. [C#高级编程]基础知识摘要一

    核心C#: 值类型存储在堆栈中,而引用类型存储在托管堆上. object类型可以用于两个目的: 可以使用object引用绑定任何子类型的对象 object类型执行许多一般用途的基本方法,包括Equal ...