23种设计模式 之 Observer模式(观察者模式)[C语言实现] 4/23
图1 Observer模式结构图
三、场景设计
当同一组数据进行分析统计时,我们希望能够提供多种形式的表示,如:以表格统计显示、以柱状图统计显示、以百分比统计显示等等。这些表示都依赖于同一组数据,当这些数据发生改变时,所有的统计显示同时也要发生变化。
四、C语言实现
分析:表格显示、柱状图显示和百分比显示相当于观察者(Observer),而统计数据相当于被观察的主题对象(Subject)。(注:使用C语言实现设计模式时,可以根据具体场景,将类理解为结构体或一组函数(组件)等)
// 类型定义(Observer和Subject)
1. 观察者
// 观察者回调typedef struct{ ... void (*update)(int); ...}observer_t;
2. 被观察者
// 观察者链表typedef struct _observer_link_t{ observer_t *observer; struct _observer_link_t *next;}observer_link_t;
// 被观察者typedef struct _subject_t{ int data; // 被观察数据 observer_link_t *observer; // 观察者链表(队列)}subject_t;
// 更新统计(观察者更新)
1. 更新 表格统计显示
void updateform(int data){...}...
2. 更新 柱状图统计显示
void updatecylinder(int data){...}...
3. 更新 百分比统计显示
void updatepercent(int data){...}...
4. 初始化观察者
void initobserver(observer_t *observer, void (*update)(int)){ ... observer.update = update; ...}...
// 统计数据(Subject接口)
// 加入观察者队列int attachobserver(subject_t *subject, observer_t *observer){ // 在subject中的observer链表中增加节点}
// 从观察者队列删除int detachobserver(subject_t *subject, observer_t *observer){ // 在subject中的observer链表中删除节点}
// 修改被观察数据int setdata(subject_t *subject, int *value){ subject->data = value;}
// 通知观察者void notifyobserver(subject_t *subject){ observer_link_t *node = subject->observer; while(NULL != node) { node->update(subject->data); } ...}
// 初始化主题对象void initsubject(subject_t *subject){ memset(subject, 0, sizeof(subject_t)); subject->observer = NULL;}
// Observer模式使用
int main(int argc, const char *agrc[]){ observer_t form; // 表格对象 observer_t cylinder; // 柱状图对象 observer_t percent; // 百分比对象 subject_t subject; // 统计数据 // 初始化观察者 initobserver(&form, updateform); initobserver(&cylinder, updatecylinder); initobserver(&percent, updatepercent); // 初始化被观察者 initsubject(&subject); // 观察者关注subject attachobserver(&subject, &form); attachobserver(&subject, &cylinder); attachobserver(&subject, &percent); // 修改状态,并通知观察者 setdata(&subject, 1); notifyobserver(&subject); setdata(&subject, 1); notifyobserver(&subject); ... return 0;}
—— 邹祁峰
2012.11.15