[C++][设计模式][观察者模式]详细讲解

avatar
作者
筋斗云
阅读量:2

目录


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

广告一刻

为您即时展示最新活动产品广告消息,让您随时掌握产品活动新动态!