[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);

}

};



声明

本文内容仅代表作者观点,或转载于其他网站,本站不以此文作为商业用途
如有涉及侵权,请联系本站进行删除
转载本站原创文章,请注明来源及作者。