• 设计模式总结


    设计模式是前辈们对代码开发经验的总结,是解决特定问题的⼀系列套路。它不是语法规定,⽽是⼀套来提⾼代码可复⽤性、可维护性、可读性、稳健性以及安全性的解决⽅案。
    六⼤原则
    单⼀职责原则
    类的职责应该单⼀,⼀个⽅法只做⼀件事,职责划分清晰了
    开闭原则
    对扩展开放,对修改封闭
    对软件实体的改动,最好⽤扩展⽽⾮修改的⽅式。
    ⾥⽒替换原则
    通俗点讲,就是只要⽗类能出现的地⽅,⼦类就可以出现,⽽且替换为⼦类也不会产⽣任何错误或异常。
    在继承类时,务必重写⽗类中所有的⽅法,尤其需要注意⽗类的protected⽅法,⼦类尽量不要暴露⾃⼰的public⽅法供外界调⽤。
    依赖倒置原则
    ⾼层模块不应该依赖低层模块,两者都应该依赖其抽象. 不可分割的原⼦逻辑就是低层模式,原⼦逻辑组装成的就是⾼层模块。
    模块间依赖通过抽象(接⼝)发⽣,具体类之间不直接依赖
    迪⽶特法则【最少知道法则】
    尽量减少对象之间的交互,从⽽减⼩类之间的耦合。⼀个对象应该对其他对象有最少的了解。
    接⼝隔离原则
    客⼾端不应该依赖它不需要的接⼝,类间的依赖关系应该建⽴在最⼩的接⼝上
    使⽤建议:接⼝设计尽量精简单⼀,但是不要对外暴露没有实际意义的接⼝。
    ⽤例:修改密码,不应该提供修改⽤⼾信息接⼝,⽽就是单⼀的最⼩修改密码接⼝,更不要暴露数据库操作

    从整体上来理解六⼤设计原则,可以简要的概括为⼀句话,⽤抽象构建框架,⽤实现扩展细节,具体  到每⼀条设计原则,则对应⼀条注意事项:

    单⼀职责原则告诉我们实现类要职责单⼀;
    ⾥⽒替换原则告诉我们不要破坏继承体系;
    依赖倒置原则告诉我们要⾯向接⼝编程;
    接⼝隔离原则告诉我们在设计接⼝的时候要精简单⼀;
    迪⽶特法则告诉我们要降低耦合;
    开闭原则是总纲,告诉我们要对扩展开放,对修改关闭。
    单例模式

    一个类只能创建一个对象,即单例模式,该模式可以保证系统中该类只有一个实例,并提供一个
    访问它的全局访问点,该实例被所有程序模块共享。
    比如在某个服务器程序中,该服务器的配置信息存放在一个文件中,这些配置数据由一个单例对象统一读取,然后服务进程中的其他对象再通过这个单例对象获取这些配置信息,这种方式简化了在复杂环境下的配置管理。
    单例模式有两种实现模式:饿汉模式,懒汉模式


    饿汉模式


    就是说不管你将来用不用,程序启动时就会创建⼀个唯⼀的实例对象。 因为单例对象已经确定, 所以⽐较适⽤于多线程环境中, 多线程获取单例对象不需要加锁, 可以有效的避免资源竞争, 提⾼性能。

    1. #include
    2. using namespace std;
    3. class Singleton {
    4. private:
    5. static Singleton _eton;
    6. int _data;
    7. private:
    8. Singleton() :_data(99) {
    9. cout << "单例对象构造" << endl;
    10. }
    11. ~Singleton() {}
    12. public:
    13. Singleton(const Singleton&) = delete;
    14. Singleton& operator=(const Singleton&) = delete;
    15. static Singleton& getInstance()
    16. {
    17. return _eton;
    18. }
    19. int getData() {
    20. return _data;
    21. }
    22. };
    23. Singleton Singleton::_eton;
    24. int main()
    25. {
    26. //cout << Singleton::getInstance().getData() << endl;
    27. return 0;
    28. }

    结果演示
    注释前:

    ce74fbe35c21468a8ac2b1750b3617b1.png

    注释后:

    bfde43b2867b405b913f0a3551347b4a.png

    由此可见,在main函数启动后,就会自动构造单例对象,即便是没有调用该单例对象。

    懒汉模式


    第⼀次使⽤要使⽤单例对象的时候创建实例对象。如果单例对象构造特别耗时或者耗费资源(加载插件、加载⽹络资源等), 可以选择懒汉模式, 在第⼀次使⽤的时候才创建对象。

    1. class B
    2. {
    3. public:
    4. static B* GetInstance()
    5. {
    6. if (_inst == nullptr)
    7. {
    8. _inst = new B;
    9. }
    10. return _inst;
    11. }
    12. static void DelInstance()
    13. {
    14. if (_inst)
    15. {
    16. delete _inst;
    17. _inst = nullptr;
    18. }
    19. }
    20. private:
    21. B()
    22. {}
    23. ~B()
    24. {
    25. cout << "数据写到文件" << endl;
    26. }
    27. B(const B& aa) = delete;
    28. B& operator=(const B& aa) = delete;
    29. static B* _inst;
    30. class gc
    31. {
    32. public:
    33. ~gc()
    34. {
    35. DelInstance();
    36. }
    37. };
    38. static gc _gc;
    39. };
    40. B* B::_inst = nullptr;
    41. B::gc B::_gc;

    下面再介绍一种《Effective C++》⼀书作者 Scott Meyers 提出的⼀种更加优雅简便的单例模式 Meyers' Singleton in C++。

    C++11 Static local variables 特性以确保C++11起,静态变量将能够在满⾜ thread-safe 的前提下唯⼀地被构造和析构。

    1. #include
    2. using namespace std;
    3. class Singleton {
    4. private:
    5. int _data;
    6. private:
    7. Singleton() :_data(99) {
    8. cout << "单例对象构造" << endl;
    9. }
    10. ~Singleton() {}
    11. public:
    12. Singleton(const Singleton&) = delete;
    13. Singleton& operator=(const Singleton&) = delete;
    14. static Singleton& getInstance()
    15. {
    16. static Singleton _eton;
    17. return _eton;
    18. }
    19. int getData() {
    20. return _data;
    21. }
    22. };
    23. int main()
    24. {
    25. cout << Singleton::getInstance().getData() << endl;
    26. return 0;
    27. }

    结果演示

    没注释:

    ca332604d967421ea05b51f59d693c59.png

    注释后:

    4dd720dc0e0549708cd62362b066de56.png

    由此可见,懒汉模式中,只有第一次创建对象时,才会构造单例对象。

    工厂模式
    ⼯⼚模式是⼀种创建型设计模式, 它提供了⼀种创建对象的最佳⽅式。在⼯⼚模式中,我们创建对象时不会对上层暴露创建逻辑,⽽是通过使⽤⼀个共同结构来指向新创建的对象,以此实现创建-使⽤的分离。
    ⼯⼚模式可以分为:简单工厂模式,工厂方法模式,抽象工厂模式。
    简单工厂模式
    简单⼯⼚模式: 简单⼯⼚模式实现由⼀个⼯⼚对象通过类型决定创建出来指定产品类的实例。假设
    有个⼯⼚能⽣产出⽔果,当客⼾需要产品的时候明确告知⼯⼚⽣产哪类⽔果,⼯⼚需要接收⽤⼾提
    供的类别信息,当新增产品的时候,⼯⼚内部去添加新产品的⽣产⽅式。
    简单⼯⼚模式:通过参数控制可以⽣产任何产品
     优点:简单粗暴,直观易懂。使⽤⼀个⼯⼚⽣产同⼀等级结构下的任意产品
     缺点:
     1. 所有东西⽣产在⼀起,产品太多会导致代码量庞⼤
     2. 开闭原则遵循(开放拓展,关闭修改)的不是太好,要新增产品就必须修改⼯⼚⽅法。
    1. #include
    2. #include
    3. #include
    4. class Fruit {
    5. public:
    6. Fruit() {}
    7. virtual void show() = 0;
    8. };
    9. class Apple : public Fruit {
    10. public:
    11. Apple() {}
    12. virtual void show() {
    13. std::cout << "我是一个苹果" << std::endl;
    14. }
    15. };
    16. class Banana : public Fruit {
    17. public:
    18. Banana() {}
    19. virtual void show() {
    20. std::cout << "我是一个香蕉" << std::endl;
    21. }
    22. };
    23. class FruitFactory {
    24. public:
    25. static std::shared_ptr create(const std::string& name) {
    26. if (name == "苹果") {
    27. return std::make_shared();
    28. }
    29. else if (name == "⾹蕉") {
    30. return std::make_shared();
    31. }
    32. return std::shared_ptr();
    33. }
    34. };
    35. int main()
    36. {
    37. std::shared_ptr fruit = FruitFactory::create("苹果");
    38. fruit->show();
    39. fruit = FruitFactory::create("⾹蕉");
    40. fruit->show();
    41. return 0;
    42. }
    运行结果:
    6fc8b8f8768c4fd88d16f4b136033509.png
    这个模式的结构和管理产品对象的⽅式⼗分简单, 但是它的扩展性⾮常差,当我们需要新增产品的时候,就需要去修改⼯⼚类新增⼀个类型的产品创建逻辑,违背了开闭原则。
    工厂方法模式
    ⼯⼚⽅法模式: 在简单⼯⼚模式下新增多个⼯⼚,多个产品,每个产品对应⼀个⼯⼚。假设现在有A、B 两种产品,则开两个⼯⼚,⼯⼚ A 负责⽣产产品 A,⼯⼚ B 负责⽣产产品 B,⽤⼾只知道产品的⼯⼚名,⽽不知道具体的产品信息,⼯⼚不需要再接收客⼾的产品类别,⽽只负责⽣产产品。

    ⼯⼚⽅法:定义⼀个创建对象的接⼝,但是由⼦类来决定创建哪种对象,使⽤多个⼯⼚分别⽣产指定的固定产品
     优点: 
     1. 减轻了⼯⼚类的负担,将某类产品的⽣产交给指定的⼯⼚来进⾏
     2. 开闭原则遵循较好,添加新产品只需要新增产品的⼯⼚即可,不需要修改原先的⼯⼚类
     缺点:对于某种可以形成⼀组产品族的情况处理较为复杂,需要创建⼤量的⼯⼚类. 

    1. #include
    2. #include
    3. #include
    4. class Fruit {
    5. public:
    6. Fruit() {}
    7. virtual void show() = 0;
    8. };
    9. class Apple : public Fruit {
    10. public:
    11. Apple() {}
    12. virtual void show() {
    13. std::cout << "我是一个苹果" << std::endl;
    14. }
    15. private:
    16. std::string _color;
    17. };
    18. class Banana : public Fruit {
    19. public:
    20. Banana() {}
    21. virtual void show() {
    22. std::cout << "我是一个香蕉" << std::endl;
    23. }
    24. };
    25. class FruitFactory {
    26. public:
    27. virtual std::shared_ptr create() = 0;
    28. };
    29. class AppleFactory : public FruitFactory {
    30. public:
    31. virtual std::shared_ptr create() {
    32. return std::make_shared();
    33. }
    34. };
    35. class BananaFactory : public FruitFactory {
    36. public:
    37. virtual std::shared_ptr create() {
    38. return std::make_shared();
    39. }
    40. };
    41. int main()
    42. {
    43. std::shared_ptr factory(new AppleFactory());
    44. std::shared_ptr fruit = factory->create();
    45. fruit->show();
    46. factory.reset(new BananaFactory());
    47. fruit = factory->create();
    48. fruit->show();
    49. return 0;
    50. }

     运行结果:

    b8aa883db0c84c5b8ca565197d541e72.png

    ⼯⼚⽅法模式每次增加⼀个产品时,都需要增加⼀个具体产品类和⼯⼚类,这会使得系统中类的个数成倍增加,在⼀定程度上增加了系统的耦合度。
    抽象工厂模式
    抽象⼯⼚模式: ⼯⼚⽅法模式通过引⼊⼯⼚等级结构,解决了简单⼯⼚模式中⼯⼚类职责太重的问
    题,但由于⼯⼚⽅法模式中的每个⼯⼚只⽣产⼀类产品,可能会导致系统中存在⼤量的⼯⼚类,势
    必会增加系统的开销。此时,我们可以考虑将⼀些相关的产品组成⼀个产品族(位于不同产品等级
    结构中功能相关联的产品组成的家族),由同⼀个⼯⼚来统⼀⽣产,这就是抽象⼯⼚模式的基本思
    想。
    抽象⼯⼚:围绕⼀个超级⼯⼚创建其他⼯⼚。每个⽣成的⼯⼚按照⼯⼚模式提供对象。
    思想:将⼯⼚抽象成两层,抽象⼯⼚ & 具体⼯⼚⼦类, 在⼯⼚⼦类种⽣产不同类型的⼦产品
    1. #include
    2. #include
    3. #include
    4. class Fruit {
    5. public:
    6. Fruit() {}
    7. virtual void show() = 0;
    8. };
    9. class Apple : public Fruit {
    10. public:
    11. Apple() {}
    12. virtual void show() {
    13. std::cout << "我是一个苹果" << std::endl;
    14. }
    15. private:
    16. std::string _color;
    17. };
    18. class Banana : public Fruit {
    19. public:
    20. Banana() {}
    21. virtual void show() {
    22. std::cout << "我是一个香蕉" << std::endl;
    23. }
    24. };
    25. class Animal {
    26. public:
    27. virtual void voice() = 0;
    28. };
    29. class Lamp : public Animal {
    30. public:
    31. void voice() { std::cout << "咩咩咩\n"; }
    32. };
    33. class Dog : public Animal {
    34. public:
    35. void voice() { std::cout << "汪汪汪\n"; }
    36. };
    37. class Factory {
    38. public:
    39. virtual std::shared_ptr getFruit(const std::string& name) = 0;
    40. virtual std::shared_ptr getAnimal(const std::string& name) = 0;
    41. };
    42. class FruitFactory : public Factory {
    43. public:
    44. virtual std::shared_ptr getAnimal(const std::string& name) {
    45. return std::shared_ptr();
    46. }
    47. virtual std::shared_ptr getFruit(const std::string& name) {
    48. if (name == "苹果") {
    49. return std::make_shared();
    50. }
    51. else if (name == "⾹蕉") {
    52. return std::make_shared();
    53. }
    54. return std::shared_ptr();
    55. }
    56. };
    57. class AnimalFactory : public Factory {
    58. public:
    59. virtual std::shared_ptr getFruit(const std::string& name) {
    60. return std::shared_ptr();
    61. }
    62. virtual std::shared_ptr getAnimal(const std::string& name) {
    63. if (name == "⼩⽺") {
    64. return std::make_shared();
    65. }
    66. else if (name == "⼩狗") {
    67. return std::make_shared();
    68. }
    69. return std::shared_ptr();
    70. }
    71. };
    72. class FactoryProducer {
    73. public:
    74. static std::shared_ptr getFactory(const std::string& name) {
    75. if (name == "动物") {
    76. return std::make_shared();
    77. }
    78. else {
    79. return std::make_shared();
    80. }
    81. }
    82. };
    83. int main()
    84. {
    85. std::shared_ptr fruit_factory = FactoryProducer::getFactory("⽔果");
    86. std::shared_ptr fruit = fruit_factory->getFruit("苹果");
    87. fruit->show();
    88. fruit = fruit_factory->getFruit("⾹蕉");
    89. fruit->show();
    90. std::shared_ptr animal_factory = FactoryProducer::getFactory("动物");
    91. std::shared_ptr animal = animal_factory->getAnimal("⼩⽺");
    92. animal->voice();
    93. animal = animal_factory->getAnimal("⼩狗");
    94. animal->voice();
    95. return 0;
    96. }

    运行结果:

    84b18cc41e5f472da8e9db09dfb4880e.png

    建造者模式
    建造者模式是⼀种创建型设计模式, 使⽤多个简单的对象⼀步⼀步构建成⼀个复杂的对象,能够将⼀个复杂的对象的构建与它的表⽰分离,提供⼀种创建对象的最佳⽅式。主要⽤于解决对象的构建过于复杂的问题。
    建造者模式主要基于四个核⼼类实现:
    抽象产品类:

    具体产品类:⼀个具体的产品对象类
    抽象Builder类:创建⼀个产品对象所需的各个部件的抽象接⼝
    具体产品的Builder类:实现抽象接⼝,构建各个部件
    指挥者Director类:统⼀组建过程,提供给调⽤者使⽤,通过指挥者来构造产品
    1. #include
    2. #include
    3. /*抽象电脑类*/
    4. class Computer {
    5. public:
    6. Computer() {}
    7. void setBoard(const std::string& board) { _board = board; }
    8. void setDisplay(const std::string& display) { _display = display; }
    9. virtual void setOs() = 0;
    10. std::string toString() {
    11. std::string computer = "Computer:{\n";
    12. computer += "\tboard=" + _board + ",\n";
    13. computer += "\tdisplay=" + _display + ",\n";
    14. computer += "\tOs=" + _os + ",\n";
    15. computer += "}\n";
    16. return computer;
    17. }
    18. protected:
    19. std::string _board;
    20. std::string _display;
    21. std::string _os;
    22. };
    23. /*具体产品类*/
    24. class MacBook : public Computer {
    25. public:
    26. MacBook() {}
    27. virtual void setOs() {
    28. _os = "Max Os X12";
    29. }
    30. };
    31. /*抽象建造者类:包含创建⼀个产品对象的各个部件的抽象接⼝*/
    32. class Builder {
    33. public:
    34. virtual void buildBoard(const std::string& board) = 0;
    35. virtual void buildDisplay(const std::string& display) = 0;
    36. virtual void buildOs() = 0;
    37. virtual std::shared_ptr build() = 0;
    38. };
    39. /*具体产品的具体建造者类:实现抽象接⼝,构建和组装各个部件*/
    40. class MacBookBuilder : public Builder {
    41. public:
    42. MacBookBuilder() : _computer(new MacBook()) {}
    43. virtual void buildBoard(const std::string& board) {
    44. _computer->setBoard(board);
    45. }
    46. virtual void buildDisplay(const std::string& display) {
    47. _computer->setDisplay(display);
    48. }
    49. virtual void buildOs() {
    50. _computer->setOs();
    51. }
    52. virtual std::shared_ptr build() {
    53. return _computer;
    54. }
    55. private:
    56. std::shared_ptr _computer;
    57. };
    58. /*指挥者类,提供给调⽤者使⽤,通过指挥者来构造复杂产品*/
    59. class Director {
    60. public:
    61. Director(Builder* builder) :_builder(builder) {}
    62. void construct(const std::string& board, const std::string& display) {
    63. _builder->buildBoard(board);
    64. _builder->buildDisplay(display);
    65. _builder->buildOs();
    66. }
    67. private:
    68. std::shared_ptr _builder;
    69. };
    70. int main()
    71. {
    72. Builder* buidler = new MacBookBuilder();
    73. std::unique_ptr pd(new Director(buidler));
    74. pd->construct("英特尔主板", "VOC显示器");
    75. std::shared_ptr computer = buidler->build();
    76. std::cout << computer->toString();
    77. return 0;
    78. }

     运行结果:

    9b5c90d666524029bff6d379d6753aba.png

    代理模式
    代理模式指代理控制对其他对象的访问,也就是代理对象控制对原对象的引⽤。在某些情况下,⼀个对象不适合或者不能直接被引⽤访问,⽽代理对象可以在客⼾端和⽬标对象之间起到中介的作⽤。
    代理模式的结构包括⼀个是真正的你要访问的对象(⽬标类),⼀个是代理对象。⽬标对象与代理对象实现同⼀个接⼝,先访问代理类再通过代理类访问⽬标对象。
    代理模式分为静态代理、动态代理:
    静态代理指的是,在编译时就已经确定好了代理类和被代理类的关系。也就是说,在编译时就已经确定了代理类要代理的是哪个被代理类。
    动态代理指的是,在运⾏时才动态⽣成代理类,并将其与被代理类绑定。这意味着,在运⾏时才能确定代理类要代理的是哪个被代理类。
    以租房为例,房东将房⼦租出去,但是要租房⼦出去,需要发布招租启⽰, 带⼈看房,负责维修,这些⼯作中有些操作并⾮房东能完成,因此房东为了图省事,将房⼦委托给中介进⾏租赁。 代理模式实现:
    1. /*房东要把⼀个房⼦通过中介租出去理解代理模式*/
    2. #include
    3. #include
    4. class RentHouse {
    5. public:
    6. virtual void rentHouse() = 0;
    7. };
    8. /*房东类:将房⼦租出去*/
    9. class Landlord : public RentHouse {
    10. public:
    11. void rentHouse() {
    12. std::cout << "将房子租出去\n";
    13. }
    14. };
    15. /*中介代理类:对租房⼦进⾏功能加强,实现租房以外的其他功能*/
    16. class Intermediary : public RentHouse {
    17. public:
    18. void rentHouse() {
    19. std::cout << "发布招租启示\n";
    20. std::cout << "带人看房\n";
    21. _landlord.rentHouse();
    22. std::cout << "负责租后维修\n";
    23. }
    24. private:
    25. Landlord _landlord;
    26. };
    27. int main()
    28. {
    29. Intermediary intermediary;
    30. intermediary.rentHouse();
    31. return 0;
    32. }

    运行结果:

    9af79b328f7246febd92e0f421f93ec9.png

  • 相关阅读:
    挂站服务器什么意思?挂站服务器可以挂多少网站?
    Android 开机流程介绍
    基于布谷鸟优化的BP神经网络(预测应用) - 附代码
    [数据集][VOC]挖掘机数据集voc格式4288张介绍
    JSON详解
    Java mysql 双数据源
    算法学习笔记2023.1
    理解李彦宏的“不卷模型,卷应用”理念
    (学习日记)2022.8.12
    CSS 取消input的上下箭头
  • 原文地址:https://blog.csdn.net/wmh_1234567/article/details/139533754