Mediator模式
一、初识中介者模式
那些年,我们一起上过的大学,班级里有班长,有团书记。想一想如果没有QQ这种通讯工具的话,那么班长或者团支书该怎样下达消息呢??同时,班级上两个同学之间也可惜沟通啊,沟通一下,院里哪个女生,哪个帅哥呀~~~如果没有QQ的话,大概就是下面的情景:
哎呀呀,看看这个乱那。如果同学的数目多起来就会变成网状的结构啦。原本把一个系统分割成一些对象是可以增强复用性的,但是现在的情况是,这些兑现之间存在着大量的联系,耦合性极高。这是很不利于复用的,同时这种情况使得系统的灵活性大大的降低,使得对系统的扩展很难,要是新转来一个学生的话,要改动的地方就多了去了。
如果现在可以使用QQ,那么可以采用另一种方式设计这个系统呢,比如做成星形的结构:
看看这种“星形结构”和“网状结构”的区别吧,显然采用星形结构就可以避免上面的网状结构存在的问题了,实际上这里的QQ就是指的是中介,这样一来每个学生对象就不用存在耦合了,同学之间需要交流可以通过一个QQ群。
本人认为,由原来的网状结构到星形结构的转变是理解中介者模式比较好的途径,下面来具体看看中介者模式,GoF说:
中介者模式(Mediator Pattern):定义一个中介对象来封装系列对象之间的交互。中介者使各个对象不需要显示地相互引用,从而使其耦合性松散,而且可以独立地改变他们之间的交互。
看看结构图的形象描述吧:
对于设计模式这个东西,理解了,应该写一个代码体会一下:看看这些类,比如:Mediator,ConcreteMediator,Colleague和ConcreteColleage1到底是怎么组织的。
View Code #include <iostream> #include <vector> #include <string> using namespace std; //抽象的同事类 class Colleage { private: string name; string content; public: Colleage(string n = " "):name(n){}; void set_name(string name) { this->name = name; } string get_name() { return this->name; } void set_content(string content) { this->content = content; } string get_content() { return content; } virtual void talk(){}; }; //具体的同事类:班长 class Monitor : public Colleage { public: Monitor(string n = ""):Colleage(n){}; virtual void talk() { cout<<"大班长 说:"<<get_content()<<endl; } }; //具体的同事类:团支书 class TuanZhiShu : public Colleage { public: TuanZhiShu(string n = ""):Colleage(n){}; virtual void talk() { cout<<"团支书 说:"<<get_content()<<endl; } }; //具体的同事类:同学A class StudentA : public Colleage { public: StudentA(string n = ""):Colleage(n){}; virtual void talk() { cout<<"学生 A 说:"<<get_content()<<endl; } }; //具体的同事类:同学B class StudentB : public Colleage { public: StudentB(string n = ""):Colleage(n){}; virtual void talk() { cout<<"学生 B 说:"<<get_content()<<endl; } }; //抽象中介者 class Mediator { public: vector<Colleage*> studentList; virtual void add_student(Colleage *student) { studentList.push_back(student); }; virtual void notify(Colleage *student){}; virtual void chart(Colleage *student1,Colleage *student2){}; }; //具体中介者qq通讯平台 class QQMediator : public Mediator { public: virtual void notify(Colleage *student) { student->talk(); for(int i = 0 ; i < studentList.size() ; ++i) { //不是说话者 if(student != studentList[i]) { studentList[i]->talk(); } } }; virtual void chart(Colleage *student1,Colleage *student2) { student1->talk(); student2->talk(); } }; int main() { QQMediator qq; Monitor studentMonitor("Vincent"); TuanZhiShu studentTuanZhiShu("Robort"); StudentA studentA("Sam"); StudentB studentB("Tom"); /*----------------------班长发通知-----------------------------*/ cout<<"下面的班长发布一个通知的场景:"<<endl; //将同学们加入到qq群中 qq.add_student(&studentMonitor); qq.add_student(&studentTuanZhiShu); qq.add_student(&studentA); qq.add_student(&studentB); //设置大家的回复信息 studentMonitor.set_content("明天下午2点开年级会,收到回复^^。"); studentTuanZhiShu.set_content("知道了,肯定到!!"); studentA.set_content("收到了,但是可能晚点到!!"); studentB.set_content("收到了,但是明天要去面试!!"); //开始发通知 qq.notify(&studentMonitor); /*--------------------两个同学私下交流--------------------------*/ cout<<endl<<"下面是两个同学的私下交流:"<<endl; studentMonitor.set_content("你觉得咱们“软件项目管理老师”讲的怎么样?"); studentA.set_content("我觉得讲的不够生动,还点名,不太好!!!"); qq.chart(&studentMonitor,&studentA); return 0; }
1 #include <iostream> 2 #include <vector> 3 #include <string> 4 5 using namespace std; 6 7 8 //抽象的同事类 9 class Colleage 10 { 11 private: 12 string name; 13 string content; 14 public: 15 Colleage(string n = " "):name(n){}; 16 void set_name(string name) 17 { 18 this->name = name; 19 } 20 string get_name() 21 { 22 return this->name; 23 } 24 void set_content(string content) 25 { 26 this->content = content; 27 } 28 string get_content() 29 { 30 return content; 31 } 32 virtual void talk(){}; 33 34 }; 35 //具体的同事类:班长 36 class Monitor : public Colleage 37 { 38 public: 39 Monitor(string n = ""):Colleage(n){}; 40 virtual void talk() 41 { 42 cout<<"大班长 说:"<<get_content()<<endl; 43 } 44 }; 45 //具体的同事类:团支书 46 class TuanZhiShu : public Colleage 47 { 48 public: 49 TuanZhiShu(string n = ""):Colleage(n){}; 50 virtual void talk() 51 { 52 cout<<"团支书 说:"<<get_content()<<endl; 53 } 54 }; 55 //具体的同事类:同学A 56 class StudentA : public Colleage 57 { 58 public: 59 StudentA(string n = ""):Colleage(n){}; 60 virtual void talk() 61 { 62 cout<<"学生 A 说:"<<get_content()<<endl; 63 } 64 }; 65 //具体的同事类:同学B 66 class StudentB : public Colleage 67 { 68 public: 69 StudentB(string n = ""):Colleage(n){}; 70 virtual void talk() 71 { 72 cout<<"学生 B 说:"<<get_content()<<endl; 73 } 74 }; 75 //抽象中介者 76 class Mediator 77 { 78 public: 79 vector<Colleage*> studentList; 80 virtual void add_student(Colleage *student) 81 { 82 studentList.push_back(student); 83 }; 84 virtual void notify(Colleage *student){}; 85 virtual void chart(Colleage *student1,Colleage *student2){}; 86 }; 87 //具体中介者qq通讯平台 88 class QQMediator : public Mediator 89 { 90 public: 91 virtual void notify(Colleage *student) 92 { 93 student->talk(); 94 for(int i = 0 ; i < studentList.size() ; ++i) 95 { 96 //不是说话者 97 if(student != studentList[i]) 98 { 99 studentList[i]->talk(); 100 } 101 } 102 }; 103 virtual void chart(Colleage *student1,Colleage *student2) 104 { 105 student1->talk(); 106 student2->talk(); 107 } 108 }; 109 110 111 int main() 112 { 113 QQMediator qq; 114 Monitor studentMonitor("Vincent"); 115 TuanZhiShu studentTuanZhiShu("Robort"); 116 StudentA studentA("Sam"); 117 StudentB studentB("Tom"); 118 /*----------------------班长发通知-----------------------------*/ 119 cout<<"下面的班长发布一个通知的场景:"<<endl; 120 //将同学们加入到qq群中 121 qq.add_student(&studentMonitor); 122 qq.add_student(&studentTuanZhiShu); 123 qq.add_student(&studentA); 124 qq.add_student(&studentB); 125 //设置大家的回复信息 126 studentMonitor.set_content("明天下午2点开年级会,收到回复^^。"); 127 studentTuanZhiShu.set_content("知道了,肯定到!!"); 128 studentA.set_content("收到了,但是可能晚点到!!"); 129 studentB.set_content("收到了,但是明天要去面试!!"); 130 //开始发通知 131 qq.notify(&studentMonitor); 132 /*--------------------两个同学私下交流--------------------------*/ 133 cout<<endl<<"下面是两个同学的私下交流:"<<endl; 134 studentMonitor.set_content("你觉得咱们“软件项目管理老师”讲的怎么样?"); 135 studentA.set_content("我觉得讲的不够生动,还点名,不太好!!!"); 136 qq.chart(&studentMonitor,&studentA); 137 return 0; 138 }
下面是运行的结果:
看看,这样就利用qq这个平台完成了同学之间的交流。并且同学之间并没有相互联系,联系都是通过qq实现的,如果现在要对象进行扩展,比如也可以通过飞信啊,只需要加入class FeiXin:public Mediator 就可以了,如果想扩展同学C只需要,class StudentC:public Colleage 即可。并且在客户端的使用也是很方便的哦~~这时回过头想一想如果不用“中介者模式”的话,系统会是什么样呢???
二、使用中介者模式的场合和优缺点
使用终结者模式的场合
1.一组定义良好的对象,现在要进行复杂的通信。
2.定制一个分布在多个类中的行为,而又不想生成太多的子类。
可以看出,中介对象主要是用来封装行为的,行为的参与者就是那些对象,但是通过中介者,这些对象不用相互知道。呵呵~~~
使用中介者模式的优点:
1.降低了系统对象之间的耦合性,使得对象易于独立的被复用。
2.提高系统的灵活性,使得系统易于扩展和维护。
使用中介者模式的缺点:
中介者模式的缺点是显而易见的,因为这个“中介“承担了较多的责任,所以一旦这个中介对象出现了问题,那么整个系统就会受到重大的影响。