当前位置:   article > 正文

C++设计模式---结构型模式

C++设计模式---结构型模式

1. 代理模式

        为其他对象提供一种代理以控制对这个对象的访问。在某些情况下,一个对象不适合或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用。

1.1 代理模式基本理论 

1.2代理模式案例 

subject(抽象主题角色):真实主题与代理主题的共同接口。
RealSubject(真实主题角色)∶定义了代理角色所代表的真实对象。
Proxy(代理主题角色):含有对真实主题角色的引用,代理角色通常在将客户端调用传递给真实主题对象之前或者之后执行某些操作,而不是单纯返回真实的对象。 

代码示例:

  1. #include <iostream>
  2. using namespace std;
  3. //提供一种代理来控制对其他对象的访问
  4. class AbstractCommonInterface
  5. {
  6. public:
  7. virtual void run() = 0;
  8. };
  9. //我已经写好的系统
  10. class MySystem:public AbstractCommonInterface
  11. {
  12. public:
  13. virtual void run()
  14. {
  15. cout << "系统启动..." << endl;
  16. }
  17. };
  18. //必须要有权限验证,提供同户名和密码
  19. class MySystemProxy :public AbstractCommonInterface
  20. {
  21. public:
  22. MySystemProxy(string username, string password)
  23. {
  24. this->mUsername = username;
  25. this->mPassword = password;
  26. pSystem = new MySystem;
  27. }
  28. bool checkUsernameAndPassword()
  29. {
  30. if (mUsername == "admin" && mPassword == "admin")
  31. {
  32. return true;
  33. }
  34. return false;
  35. }
  36. virtual void run()
  37. {
  38. if (checkUsernameAndPassword())
  39. {
  40. cout << "用户名和密码正确,验证通过..." << endl;
  41. this->pSystem->run();
  42. }
  43. else
  44. {
  45. cout << "用户名或密码错误,权限不足...." << endl;
  46. }
  47. }
  48. ~MySystemProxy() {
  49. if (pSystem != NULL)
  50. {
  51. delete pSystem;
  52. }
  53. }
  54. public:
  55. MySystem* pSystem;
  56. string mUsername;
  57. string mPassword;
  58. };
  59. void test08()
  60. {
  61. MySystemProxy* proxy = new MySystemProxy("root", "admin");
  62. proxy->run();
  63. }
  64. int main()
  65. {
  66. test08();
  67. return 0;
  68. }

2. 外观模式

        外观模式就是将复杂的子类系统抽象到同一个的接口进行管理,外界只需要通过此接口与子类系统进行交互,而不必要直接与复杂的子类系统进行交互

2.1 外观模式基本理论 

        根据迪米特法则,如果两个类不必彼此直接通信,那么这两个类就不应当发生直接的相互作用。
        Facade模式也叫外观模式,是由GoF提出的23种设计模式中的一种。Facade模式为一组具有类似功能的类群,比如类库,子系统等等,提供一个一致的简单的界面。这个一致的简单的界面被称作facade。

2.2 外观模式案例

根据类图,实现家庭影院外观模式应用。
实现KTV模式∶电视打开,灯关掉,音响打开,麦克风打开,dvd打开。

实现游戏模式:电视打开,音响打开,游戏机打开。

代码示例: 

  1. #include <iostream>
  2. using namespace std;
  3. //电视机
  4. class Televison
  5. {
  6. public:
  7. void On()
  8. {
  9. cout<<"电视机打开..." << endl;
  10. }
  11. void Off()
  12. {
  13. cout << "电视机关闭..." << endl;
  14. }
  15. };
  16. //灯
  17. class Light
  18. {
  19. public:
  20. void On()
  21. {
  22. cout << "灯打开..." << endl;
  23. }
  24. void Off()
  25. {
  26. cout << "灯关闭" << endl;
  27. }
  28. };
  29. //音箱
  30. class Audio
  31. {
  32. public:
  33. void On()
  34. {
  35. cout << "音箱打开..." << endl;
  36. }
  37. void Off()
  38. {
  39. cout << "音箱关闭" << endl;
  40. }
  41. };
  42. //麦克风
  43. class Microphone
  44. {
  45. public:
  46. void On()
  47. {
  48. cout << "麦克风打开..." << endl;
  49. }
  50. void Off()
  51. {
  52. cout << "麦克风关闭" << endl;
  53. }
  54. };
  55. //DVD
  56. class DVDPlayer
  57. {
  58. public:
  59. void On()
  60. {
  61. cout << "DVD播放器打开..." << endl;
  62. }
  63. void Off()
  64. {
  65. cout << "DVD播放器关闭" << endl;
  66. }
  67. };
  68. //游戏机
  69. class Gamemachine
  70. {
  71. public:
  72. void On()
  73. {
  74. cout << "游戏机打开..." << endl;
  75. }
  76. void Off()
  77. {
  78. cout << "游戏机关闭" << endl;
  79. }
  80. };
  81. //KTV模式
  82. class KTVModel
  83. {
  84. public:
  85. KTVModel()
  86. {
  87. pTv = new Televison;
  88. pLight = new Light;
  89. pAudio = new Audio;
  90. pMicrophone = new Microphone;
  91. pDVD = new DVDPlayer;
  92. }
  93. void OnKtv()
  94. {
  95. pTv->On();
  96. pLight->Off();
  97. pAudio->On();
  98. pMicrophone->On();
  99. pDVD->On();
  100. }
  101. void OffKtv()
  102. {
  103. pTv->Off();
  104. pLight->On();
  105. pAudio->Off();
  106. pMicrophone->Off();
  107. pDVD->Off();
  108. }
  109. ~KTVModel()
  110. {
  111. delete pTv;
  112. delete pLight;
  113. delete pAudio;
  114. delete pMicrophone;
  115. delete pDVD;
  116. }
  117. public:
  118. Televison* pTv;
  119. Light* pLight;
  120. Audio* pAudio;
  121. Microphone* pMicrophone;
  122. DVDPlayer* pDVD;
  123. };
  124. void test09()
  125. {
  126. KTVModel* ktv = new KTVModel;
  127. ktv->OnKtv();
  128. }
  129. int main()
  130. {
  131. test09();
  132. return 0;
  133. }

3. 适配器模式 

3.1 适配器模式基本理论

        将一个类的接口转换成客户希望的另外一个接口。使得原本由于接口不兼容而不能一起工作的那些类可以一起工作。 

3.2 适配器模式案例

代码示例:

  1. #include <iostream>
  2. #include<vector>
  3. #include<algorithm>
  4. using namespace std;
  5. //适配器模式就是将写好的接口(但不符合需求)转换成目标接口
  6. //这函数我已经写好
  7. struct Myprint
  8. {
  9. void operator()(int v1, int v2)
  10. {
  11. cout << v1 + v2 << endl;
  12. }
  13. };
  14. //定义目标接口 我要是配偶 适配成什么样的,
  15. //要适配成只能传一个参数的,适配for_each第三个参数的适用
  16. class Target
  17. {
  18. public:
  19. virtual void operator()(int v) = 0;
  20. };
  21. //写适配器
  22. class Adapater :public Target
  23. {
  24. public:
  25. Adapater(int param)
  26. {
  27. this->param = param;
  28. }
  29. virtual void operator() (int v)
  30. {
  31. print(v, param);
  32. }
  33. public:
  34. Myprint print;
  35. int param;
  36. };
  37. //MyBind2nd,原来param固定的10,现在提供一个方法改
  38. Adapater MyBind2nd(int v)
  39. {
  40. return Adapater(v);
  41. }
  42. int main()
  43. {
  44. vector<int> v;
  45. for (int i = 0; i < 10; i++)
  46. {
  47. v.push_back(i);
  48. }
  49. //适配器模式的运用
  50. //for_each()的第三个参数是个带一个参数的函数,但是Myprint需要两个参数
  51. for_each(v.begin(), v.end(), MyBind2nd(10));
  52. return 0;
  53. }

4. 装饰器模式

4.1 装饰器模式理论

        装饰模式又叫包装模式,通过一种对客户端透明的方式来扩展对象功能,是继承关系的一种替代。
        装饰模式就是把要附加的功能分别放在单独的类中,并让这个类包含它要装饰的对象,当需要执行时,客户端就可以有选择的、按顺序的使用装饰功能包装对象。

4.2 装饰器模式案例

代码示例: 

  1. #include <iostream>
  2. using namespace std;
  3. //一般情况下,用继承实现类的功能拓展
  4. //装饰模式 可以动态给一个类增加功能
  5. //抽象英雄
  6. class AbstractHero
  7. {
  8. public:
  9. virtual void ShowStatus() = 0;
  10. public:
  11. int mHp;
  12. int mMp;
  13. int mAt;
  14. int mDf;
  15. };
  16. //具体英雄
  17. class HeroA :public AbstractHero
  18. {
  19. public:
  20. HeroA()
  21. {
  22. mHp = 0;
  23. mMp = 0;
  24. mAt = 0;
  25. mDf = 0;
  26. }
  27. virtual void ShowStatus()
  28. {
  29. cout << "血量:" << mHp << endl;
  30. cout << "魔法:" << mMp << endl;
  31. cout << "攻击:" << mAt << endl;
  32. cout << "防御:" << mDf << endl;
  33. }
  34. };
  35. //英雄穿上某个装饰物 那么他还是个英雄
  36. //装饰物
  37. class AbstractEquipmet : public AbstractHero
  38. {
  39. public:
  40. AbstractEquipmet(AbstractHero* hero)
  41. {
  42. this->pHero = hero;
  43. }
  44. virtual void ShowStatus() = 0;
  45. public:
  46. AbstractHero* pHero;
  47. };
  48. //狂徒
  49. class KuangtuEquipment :public AbstractEquipmet
  50. {
  51. public:
  52. KuangtuEquipment(AbstractHero* hero) :AbstractEquipmet(hero) {}
  53. //增加额外的功能
  54. void AddKuangtu()
  55. {
  56. cout << "英雄穿上狂徒之后..." << endl;
  57. this->mHp = this->pHero->mHp;
  58. this->mMp = this->pHero->mMp;
  59. this->mAt = this->pHero->mAt;
  60. this->mDf = this->pHero->mDf + 30;
  61. delete this->pHero;
  62. }
  63. virtual void ShowStatus()
  64. {
  65. AddKuangtu();
  66. cout << "血量:" << mHp << endl;
  67. cout << "魔法:" << mMp << endl;
  68. cout << "攻击:" << mAt << endl;
  69. cout << "防御:" << mDf << endl;
  70. }
  71. };
  72. //无尽
  73. class Wujing : public AbstractEquipmet
  74. {
  75. public:
  76. Wujing(AbstractHero* hero) :AbstractEquipmet(hero) {}
  77. //增加额外的功能
  78. void AddWujing()
  79. {
  80. cout << "英雄穿上无尽之后..." << endl;
  81. this->mHp = this->pHero->mHp;
  82. this->mMp = this->pHero->mMp;
  83. this->mAt = this->pHero->mAt + 80;
  84. this->mDf = this->pHero->mDf;
  85. delete this->pHero;
  86. }
  87. virtual void ShowStatus()
  88. {
  89. AddWujing();
  90. cout << "血量:" << mHp << endl;
  91. cout << "魔法:" << mMp << endl;
  92. cout << "攻击:" << mAt << endl;
  93. cout << "防御:" << mDf << endl;
  94. }
  95. };
  96. void test15()
  97. {
  98. AbstractHero* hero = new HeroA;
  99. hero->ShowStatus();
  100. cout << "----------------------------" << endl;
  101. //给裸奔的英雄穿上衣服后
  102. hero = new KuangtuEquipment(hero);
  103. hero->ShowStatus();
  104. cout << "----------------------------" << endl;
  105. //装备武器
  106. hero = new Wujing(hero);
  107. hero->ShowStatus();
  108. }
  109. int main()
  110. {
  111. test15();
  112. return 0;
  113. }
声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/一键难忘520/article/detail/823085
推荐阅读
相关标签
  

闽ICP备14008679号