[C++][设计模式][观察者模式]详细讲解
DieSnowK 2024-07-01 10:05:03 阅读 58
目录
1.动机2.模式定义3.要点总结4.代码感受1.代码一1.FileSplitter.cpp2.MainForm.cpp
2.代码二1.FileSplitter.cpp2.MainForm.cpp
1.动机
在软件构建过程中,需要为某些对象建立一种“通知依赖关系”
一个对象(目标对象)的状态发生改变,所有的依赖对象(观察者对象)都将得到通知如果这样的依赖关系过于紧密, 将使软件不能很好地抵御变化 使用面向对象技术,可以将这种依赖关系弱化,并形成一种稳定的依赖关系。从而实现软件体系结构的松耦合
2.模式定义
定义对象间的一种一对多(变化)的依赖关系,以便当一个对象(Subject)的状态发生改变时,所有依赖于它的对象都得到通知并自动更新结构
3.要点总结
使用面向对象的抽象,Observer模式使得我们可以独立地改变目标与观察者,从而使二者之间的依赖关系达致松耦合目标发送通知时,无需指定观察者,通知(可以携带通知信息作为参数)会自动传播观察者自己决定是否需要订阅通知,目标对象对此一无所知Observer模式是基于事件的UI框架中非常常用的设计模式,也是MVC模式的一个重要组成部分
4.代码感受
1.代码一
1.FileSplitter.cpp
class FileSplitter
{
string m_filePath;
int m_fileNumber;
ProgressBar* m_progressBar;
public:
FileSplitter(const string& filePath, int fileNumber, ProgressBar* progressBar) :
m_filePath(filePath),
m_fileNumber(fileNumber),
m_progressBar(progressBar)
{ }
void split()
{
//1.读取大文件
//2.分批次向小文件中写入
for (int i = 0; i < m_fileNumber; i++)
{
//...
float progressValue = (i + 1) / m_fileNumber;
m_progressBar->setValue(progressValue);
}
}
};
2.MainForm.cpp
class MainForm : public Form
{
TextBox* txtFilePath;
TextBox* txtFileNumber;
ProgressBar* progressBar;
public:
void Button1_Click()
{
string filePath = txtFilePath->getText();
int number = atoi(txtFileNumber->getText().c_str());
FileSplitter splitter(filePath, number, progressBar);
splitter.split();
}
};
2.代码二
1.FileSplitter.cpp
class IProgress
{
public:
virtual void DoProgress(float value) = 0;
virtual ~IProgress(){ }
};
class FileSplitter
{
string m_filePath;
int m_fileNumber;
List<IProgress*> m_iprogressList; // 抽象通知机制,支持多个观察者
public:
FileSplitter(const string& filePath, int fileNumber) :
m_filePath(filePath),
m_fileNumber(fileNumber)
{ }
void split()
{
//1.读取大文件
//2.分批次向小文件中写入
for (int i = 0; i < m_fileNumber; i++)
{
//...
float progressValue = (i + 1) / m_fileNumber;
onProgress(progressValue); //发送通知
}
}
void addIProgress(IProgress* iprogress)
{
m_iprogressList.push_back(iprogress);
}
void removeIProgress(IProgress* iprogress)
{
m_iprogressList.remove(iprogress);
}
protected:
virtual void onProgress(float value)
{
List<IProgress*>::iterator itor = m_iprogressList.begin();
while (itor != m_iprogressList.end())
{
(*itor)->DoProgress(value); //更新进度条
itor++;
}
}
};
2.MainForm.cpp
class ConsoleNotifier : public IProgress
{
public:
virtual void DoProgress(float value)
{
cout << ".";
}
};
// 此处IProgess是接口继承
class MainForm : public Form, public IProgress
{
TextBox* txtFilePath;
TextBox* txtFileNumber;
ProgressBar* progressBar;
public:
void Button1_Click()
{
string filePath = txtFilePath->getText();
int number = atoi(txtFileNumber->getText().c_str());
FileSplitter splitter(filePath, number);
ConsoleNotifier cn;
splitter.addIProgress(this); //订阅通知
splitter.addIProgress(&cn); //订阅通知
splitter.split();
}
virtual void DoProgress(float value)
{
progressBar->setValue(value);
}
};
声明
本文内容仅代表作者观点,或转载于其他网站,本站不以此文作为商业用途
如有涉及侵权,请联系本站进行删除
转载本站原创文章,请注明来源及作者。