观察者模式C和C++语言实现
原创
©著作权归作者所有:来自51CTO博客作者觉皇不秃头的原创作品,请联系作者获取转载授权,否则将追究法律责任
观察者模式C和C++语言实现
目录
模式介绍
现在假设有一组原始数据,会有折现图、sheet表以及比例图等三种表现形式,观察者模型的思想就是在原始数据有变化后发布一条通知,三种表现形式会同步更新显示,原始数据就是Subject,三种表现形式就是Observer,下面使用C语言实现该模型。
C代码使用了链表,使用方法见我的另一篇博客:【C语言】Linux内核链表移植到应用程序中及使用方法
本章代码下载地址:https://github.com/ankun6/Observer
C实现
对于C来说因为没有类的概念,所以可以使用结构体来模拟类,使用XXXX_Init
或XXXX_Constructor
类似于这样的函数来当作构造函数,使用XXXX_Destructory
类似的函数当作析构函数进行内存释放等操作,结构体内预先定义函数指针,在初始化函数中与具体实现函数进行绑定,从而达到使用结构体调用函数的功能。
首先,对于Subject类有添加、删除、通知等操作,因为C语言中没有容器,所以使用链表进行管理,我使用的这个链表是从linux内核源码中移植出来的,使用方法可以参考上面的博客链接。
Subject类实现:
对于观察者类,首先有一个基类Observer,内部含有一个update函数用于更新,然后有继承于Observer的Chart、Sheet、Scale三个数据表现形式类,在这三个类初始化的时候对update函数指针赋值不同的函数,这样就模拟了重载。
Observer类实现:
Sheet类实现:
Chart类实现:
Scale类实现:
最后,编写测试代码进行测试:
/*
* main.c
*
* Created on: 2020年9月21日
* Author: hello
*/
#include "subject.h"
#include "chart.h"
#include "scale.h"
#include "sheet.h"
int main(void)
{
struct Subject subject;
struct Chart chart;
struct Scale scale;
struct Sheet sheet;
// 初始化基础数据
Subject_Init(&subject);
// 初始化观察者
Chart_Init(&chart);
Scale_Init(&scale);
Sheet_Init(&sheet);
// 给基础数据添加观察者对象
subject.add(&subject, &chart.parant);
subject.add(&subject, &scale.parant);
subject.add(&subject, &sheet.parant);
// 模拟数据改变
subject.value = 7281;
// 通知数据改变
subject.notify(&subject);
// 移除观察者scale
subject.remove(&subject, &scale.parant);
// 模拟数据改变
subject.value = 1234;
// 通知数据改变
subject.notify(&subject);
return 0;
}
程序运行结果:
C++实现
使用C++实现就比较容易了,下面看具体代码。
Subject类实现:
Observer基类实现:
Sheet类实现:
Chart类实现:
Scale类实现:
最后,编码测试:
#include <iostream>
#include "Subject.h"
#include "Chart.h"
#include "Sheet.h"
#include "Scale.h"
using namespace std;
int main()
{
Subject subject;
Chart chart;
Sheet sheet;
Scale scale;
// 添加观察者对象
subject.add(&chart);
subject.add(&sheet);
subject.add(&scale);
// 模拟数据改变
subject.setValue(31223);
// 通知观察者们
subject.notify();
// 移除观察者sheet
subject.remove(&sheet);
// 模拟数据改变
subject.setValue(77345);
// 通知观察者们
subject.notify();
return 0;
}
程序运行结果:
ends