obj-c编程18:多对多的观察者模式

    我们知道使用委托的设计模式可以实现一对一的通知关系,但是如果需要通知多个观察者状态变化又该如何呢?此时,需要实现观察者模式之类的内容,而不是实现委托者一对一的模式。

    观察者模式定义了一个对象可以将另一个对象注册成自身观察者的模式,对象被注册成观察者后,任何观察者关注者事件都会在其发生时send给观察者。obj-c中实现观察者模式是通过NSNotificationCenter类实现的。该类为观察者对象和事件提供了一个全局调度系统。观察者可以向其注册观测系统中特定的事件;而被观察对象,在事件发生时,也可以发布通知到NSNotificationCenter,这样任何被观察者对象的通知可以发送到任何观察者对象,从而达到多对多关系的观察者模式。

    为了完成观察者模式,一般要完成以下几个步骤:

1: 观察者使用NSNotificationCenter实例方法-addObserver将自身注册为特定事件的观察者;

2: 观察者要实现步骤1中注册的回调方法;

3: 被观察者对象要使用-postNotifiationName之类的发送方法发送消息;

4: 观察者最终要确保在释放时使用-removeObserver:将自己从NSNotificationCenter中移除。

下面用杜撰的代码来看看实际如何实现观察者模式。代码中狗仔队对于明星无聊中的对话可谓是八卦之极,star的每一句话都要设法得到啊。不过从实际代码看来,貌似明星也有意的想把这些对话内容传播出去啊!如果不想,那就类似于<<obj-c编程17:键值观察>>里的键值观察模式了哦。下面上代码:

#import <Foundation/Foundation.h>

#define msg(...) NSLog(__VA_ARGS__)

#define NOTIFICATION_NAME @"MY_NOTIFICATION"

@interface Star:NSObject{
	NSString *name;
}
	@property(readonly) NSString *name;
	-(void)talk:(NSString *)blabla;
@end

@implementation Star
	@synthesize name;

	-(id)initWithName:(NSString *)name_v{
		self = [super init];
		if(self){
			name = name_v;
		}
		return self;
	}

	-(void)talk:(NSString *)blabla{
		msg(@"%@ to say :%@",name,blabla);

		NSDictionary *secret = [NSDictionary dictionaryWithObject:blabla \
			forKey:@"words"];

		[[NSNotificationCenter defaultCenter] postNotificationName:NOTIFICATION_NAME \
			object:self userInfo:secret];
	}
@end

@interface Dog:NSObject
	-(void)broadcast:(NSNotification *)note;
@end

@implementation Dog
	-(id)init{
		self = [super init];
		if(self){
			[[NSNotificationCenter defaultCenter] addObserver:self \
				selector:@selector(broadcast:) name:NOTIFICATION_NAME \
				object:nil];
		}
		return self;
	}

	-(void)broadcast:(NSNotification *)note{
		NSString *star_name = [[note object] name];
		NSString *words = [[note userInfo] objectForKey:@"words"];
		msg(@"star %@ sayed \"%@\"",star_name,words);
	}

	-(void)dealloc{
		[[NSNotificationCenter defaultCenter] removeObserver:self];
		//[super dealloc];
	}
@end

int main(int argc,char *argv[])
{
	@autoreleasepool{
		Star *star0 = [[Star alloc] initWithName:@"lucy"];
		Star *star1 = [[Star alloc] initWithName:@"jack"];
		Dog *dog = [[Dog alloc] init];

		[star0 talk:@"no shit!"];
		[star1 talk:@"that's right!"];
	}
	return 0;
}

运行结果是我们可以预料到的:

apple@kissAir: objc_src$./9

2014-08-03 21:05:11.830 9[910:507] lucy to say :no shit!

2014-08-03 21:05:11.831 9[910:507] star lucy sayed "no shit!"

2014-08-03 21:05:11.832 9[910:507] jack to say :that's right!

2014-08-03 21:05:11.832 9[910:507] star jack sayed "that's right!"

有人会说了,你这是1对多关系哦。多对多关系也很简单,狗仔队不止一只啊,我们再来创建1个,改变的代码如下:

int main(int argc,char *argv[])
{
	@autoreleasepool{
		Star *star0 = [[Star alloc] initWithName:@"lucy"];
		Star *star1 = [[Star alloc] initWithName:@"jack"];
		Dog *dog0 = [[Dog alloc] init];
		Dog *dog1 = [[Dog alloc] init];

		[star0 talk:@"no shit!"];
		[star1 talk:@"that's right!"];
	}
	return 0;
}

运行结果如下:

apple@kissAir: objc_src$./9

2014-08-03 21:32:43.513 9[973:507] lucy to say :no shit!

2014-08-03 21:32:43.514 9[973:507] star lucy sayed "no shit!"

2014-08-03 21:32:43.515 9[973:507] star lucy sayed "no shit!"

2014-08-03 21:32:43.515 9[973:507] jack to say :that's right!

2014-08-03 21:32:43.516 9[973:507] star jack sayed "that's right!"

2014-08-03 21:32:43.516 9[973:507] star jack sayed "that's right!"

时间: 2024-09-27 10:54:05

obj-c编程18:多对多的观察者模式的相关文章

多线程编程(18)

前面的例子都是让若干线程做同样的事情, 下面这个例子中的三个线程将分别在三个画板上随机画不同颜色的椭圆. 接下来的很多事情我想要基于这个例子来做. 本例效果图: 代码文件: unit Unit1; interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms, Dialogs, StdCtrls, ExtCtrls; type TForm1 = class(TForm) Paint

Hibernate之关联关系映射(一对多和多对一映射,多对多映射)

~~~接着之前的Hibernate框架接着学习(上篇面试过后发现真的需要学习一下框架了,不然又被忽悠让去培训.)~~~ 1:Hibernate的关联映射,存在一对多和多对一映射,多对多映射: 1.1:一对多和多对一映射,举例说明: 学生和老师: 一个老师可以教多个学生 [一对多映射] 多个学生可以被一个老师教[多对一映射] 部门与员工: 一个部门有多个员工[一对多映射] 多个员工属于一个部门[多对一映射] 1.2:多对多,举例说明: 项目和开发员工:[双向一对多即多对多映射] 一个项目有多个开发

104_《Delphi5开发人员指南》

<Delphi5开发人员指南> Delphi 教程 系列书籍 (104) <Delphi5开发人员指南> 网友(邦)整理 EMail: shuaihj@163.com 下载地址: Part1 Part2 原书名: Delphi 5开发人员指南 作者: (美)Steve Teixeira.Xavier Pacheco著 出版社: 机械工业出版社 书号: 7111080408 出版日期:2000年7月 开本: 787*1092 1/16 页码: 1263 版次: 2000年7月第一版第

面向对象理论

面向对象编程举例 初步实现(面向过程的方式) 函数封装 对象封装 创建对象的方式 字面量 js var obj = { key: value, key1: value1 }; //$().css({}) //$.ajax({}) //复用性差 2. 内置构造函数 js var obj = new Object(); obj.key = value; obj.key1 = value1; //复用性差 3. 自定义构造函数 ```js function Person(){ this.key = v

Python对象的深拷贝和浅拷贝详解_python

本文内容是在<Python核心编程2>上看到的,感觉很有用便写出来,给大家参考参考! 浅拷贝 首先我们使用两种方式来拷贝对象,一种是切片,另外一种是工厂方法.然后使用id函数来看看它们的标示符 复制代码 代码如下: # encoding=UTF-8   obj = ['name',['age',18]] a=obj[:] b=list(obj) for x in obj,a,b:     print id(x)   35217032 35227912 29943304   他们的id都不同,按

《Node.js区块链开发》导读

Preface 前 言 本书原名<Nodejs开发加密货币>,大部分已经通过网络免费分享(网络上仍有原文),正式出版的时候改成了现在的名字,部分章节也做了调整.写作本书的时候,"区块链"这个称呼并不流行,但讨论的内容实际上就是区块链技术.就在本书整理出版的时候,国家发布了<中国区块链技术和应用发展白皮书>,整个社区也接纳了"区块链"这个称谓,因此改为现在的名字. "每一件与众不同的绝世好东西,其实都是以无比寂寞的勤奋为前提的,要么是

《Node.js区块链开发》一导读

前 言 本书原名<Nodejs开发加密货币>,大部分已经通过网络免费分享(网络上仍有原文),正式出版的时候改成了现在的名字,部分章节也做了调整.写作本书的时候,"区块链"这个称呼并不流行,但讨论的内容实际上就是区块链技术.就在本书整理出版的时候,国家发布了<中国区块链技术和应用发展白皮书>,整个社区也接纳了"区块链"这个称谓,因此改为现在的名字. "每一件与众不同的绝世好东西,其实都是以无比寂寞的勤奋为前提的,要么是血,要么是汗,要

数据结构Java实现07----队列:顺序队列&amp;顺序循环队列、链式队列、顺序优先队列

一.队列的概念: 队列(简称作队,Queue)也是一种特殊的线性表,队列的数据元素以及数据元素间的逻辑关系和线性表完全相同,其差别是线性表允许在任意位置插入和删除,而队列只允许在其一端进行插入操作在其另一端进行删除操作. 队列中允许进行插入操作的一端称为队尾,允许进行删除操作的一端称为队头.队列的插入操作通常称作入队列,队列的删除操作通常称作出队列. 下图是一个依次向队列中插入数据元素a0,a1,...,an-1后的示意图: 上图中,a0是当前 队头数据元素,an-1是当前 队尾数据元素. 为了

common-pools源码分析

背景   前段时间研究了下dbcp的一些源码,发现dbcp对common pools依赖比较严重,基本就是基于pool的扩展接口实现的.所以也就顺便看了下pools的源码.   总体结构   common pools的代码总体结构来说是比较简单的.   先上一个类图:     核心接口类: ObjectPool             对象池 ObjectPoolFactory   池的维护工厂 PoolableObjectFactory   可池化对象的维护工厂 KeyedObjectPool