zoukankan      html  css  js  c++  java
  • 设计模式大全

    强推一个学设计模式的网站:
    Refactoring Guru (https://refactoringguru.cn

    这是国外的一个网站,建立的初衷,是帮助开发者以更为简单便捷的方式,迅速掌握各个设计模式的学习。
    每个设计模式都对应配有一篇图文并茂的文章,让开发者可以很轻松的搞懂每个设计模式的具体工作原理,而且在每篇文章里面,均有提供对应的不同语言代码示例。

    真的是‘一站在手,天下我有’~

    在学习设计模式的时候,需要可以看懂UML图,UML(统一建模语言)https://www.w3cschool.cn/uml_tutorial/uml_tutorial-c1gf28pd.html、https://blog.csdn.net/AllenWells/article/details/47398091

    设计模式是什么?

    设计模式是软件设计中常见问题的典型解决方案。 它们就像能根据需求进行调整的预制蓝图, 可用于解决代码中反复出现的设计问题。

    设计模式与方法或库的使用方式不同, 你很难直接在自己的程序中套用某个设计模式。 模式并不是一段特定的代码, 而是解决特定问题的一般性概念。 你可以根据模式来实现符合自己程序实际所需的解决方案。

    人们常常会混淆模式和算法, 因为两者在概念上都是已知特定问题的典型解决方案。 但算法总是明确定义达成特定目标所需的一系列步骤, 而模式则是对解决方案的更高层次描述。 同一模式在两个不同程序中的实现代码可能会不一样。

    算法更像是菜谱: 提供达成目标的明确步骤。 而模式更像是蓝图: 你可以看到最终的结果和模式的功能, 但需要自己确定实现步骤。

    模式包含哪些内容?

    大部分模式都有正规的描述方式, 以便在不同情况下使用。 模式的描述通常会包括以下部分:

    • 意图部分简单描述问题和解决方案。
    • 动机部分将进一步解释问题并说明模式会如何提供解决方案。
    • 结构部分展示模式的每个部分和它们之间的关系。
    • 在不同语言中的实现提供流行编程语言的代码, 让读者更好地理解模式背后的思想。

    部分模式介绍中还列出其他的一些实用细节, 例如模式的适用性、 实现步骤以及与其他模式的关系。

    为什么我需要学习模式?

    或许你从事程序开发工作多年, 却完全不知道单例模式是什么。 很多人都是这样。 即使如此, 你可能在不自知的情况下实现过一些模式。 所以为什么不花些时间来学习它们呢?

    • 设计模式是针对软件设计中常见问题的工具箱, 其中的工具就是各种经过实践验证的解决方案。 即使你从未遇到过这些问题, 了解模式仍然非常有用, 因为它能指导你如何使用面向对象的设计原则来解决各种问题。
    • 设计模式定义了一种让你和团队成员能够更高效沟通的通用语言。 你只需说 “哦, 这里用单例就可以了”, 所有人都会理解这条建议背后的想法。 只要知晓模式及其名称, 你就无需解释什么是单例。

    关于模式的争议

    设计模式自其诞生之初似乎就饱受争议, 所以让我们来看看针对模式的最常见批评吧。

    一种针对不完善编程语言的蹩脚解决方案

    通常当所选编程语言或技术缺少必要的抽象功能时, 人们才需要设计模式。 在这种情况下, 模式是一种可为语言提供更优功能的蹩脚解决方案。

    例如, 策略模式在绝大部分现代编程语言中可以简单地使用匿名 (lambda) 函数来实现。

    低效的解决方案

    模式试图将已经广泛使用的方式系统化。 许多人会将这样的统一化认为是某种教条, 他们会 “全心全意” 地实施这样的模式, 而不会根据项目的实际情况对其进行调整。

    不当使用

    如果你只有一把铁锤, 那么任何东西看上去都像是钉子。

    这个问题常常会给初学模式的人们带来困扰: 在学习了某个模式后, 他们会在所有地方使用该模式, 即便是在较为简单的代码也能胜任的地方也是如此。

    模式的分类

    不同设计模式在其复杂程度、 细节层次以及在整个系统中的应用范围等方面各不相同。 我喜欢将其类比于道路的建造: 如果你希望让十字路口更加安全, 那么可以安装一些交通信号灯, 或者修建包含行人地下通道在内的多层互通式立交桥。

    最基础的、 底层的模式通常被称为惯用技巧。 这类模式一般只能在一种编程语言中使用。

    最通用的、 高层的模式是构架模式。 开发者可以在任何编程语言中使用这类模式。 与其他模式不同, 它们可用于整个应用程序的架构设计。

    此外, 所有模式可以根据其意图或目的来分类。 三种主要的模式类别:

    • 创建型模式提供创建对象的机制, 增加已有代码的灵活性和可复用性。
    • 结构型模式介绍如何将对象和类组装成较大的结构, 并同时保持结构的灵活和高效。
    • 行为模式负责对象间的高效沟通和职责委派。

    常见问题

    工厂模式比较

    本文将对下列概念之间的差异进行说明:

    • 工厂
    • 构建方法
    • 静态构建 (或工厂) 方法
    • 简单工厂
    • 工厂方法
    • 抽象工厂

    你可以在网上找到这些术语的参考信息。 尽管它们看上去相似, 但其含义都不一样。 许多人没有意识到这一点, 从而出现了混淆和误解。因此让我们搞清楚其中的不同之处, 一劳永逸地解决这个问题。

    1. 工厂

    工厂是一个含义模糊的术语, 表示可以创建一些东西的函数、 方法或类。 最常见的情况下, 工厂创建的是对象。 但是它们也可以创建文件和数据库记录等其他东西。

    例如, 下面这些东西都可以非正式地被称为 “工厂”:

    • 创建程序 GUI 的函数或方法;
    • 创建用户的类;
    • 以特定方式调用类构造函数的静态方法。
    • 一种创建型设计模式。

    当某人说到 “工厂” 这个词时, 其具体含义通常可以根据上下文来确定。 但如果你有疑问, 可以直接提问。 毕竟作者本人有时候也没有搞清楚。

    2. 构建方法

    构建方法在 《重构与模式》 中被定义为 “创建对象的方法”。 这意味着每个工厂方法模式的结果都是 “构建方法”, 但反过来则并非如此。 这也意味着你可以用 “构建方法” 来替代马丁·福勒在重构中使用的 “工厂方法” 和乔斯华·布洛克在 《Effective Java》 中使用的 “静态工厂方法”。

    在实际中, 构建方法只是构造函数调用的封装器。 它可能只是一个能更好地表达意图的名称。 此外, 它可以让你的代码独立于构造函数的改动, 甚至还可以包含一些特殊的逻辑, 返回已有对象以而不是创建新对象。

    许多人会仅仅因为这些方法创建了新对象而称之为 “工厂方法”。 其中的逻辑很直接: 所有的工厂都会创建对象, 而该方法会创建对象, 所以显然它是一个工厂方法。 当遇到真正的工厂方法时, 这自然会造成许多混淆。

    在下面的示例中, ​ next 是一个构建方法:

    class Number {
        private $value;
    
        public function __construct($value) {
            $this->value = $value;
        }
    
        public function next() {
            return new Number ($this->value + 1);
        }
    }
    

    3. 静态构建方法

    静态构建方法是被声明为 static的构建方法。 换句话说, 你无需创建对象就能在某个类上调用该方法。

    不要因为某些人将这些方法称为 “静态工厂方法” 而被其迷惑。 这种称呼只是一个坏习惯。 工厂方法是一种依赖于继承的设计模式。 如果将它设置为 static , 你就不能在子类中对其进行扩展, 这就破坏了该模式的目的。

    当静态构建方法返回一个新对象时, 它就成为了构造函数的替代品。

    在下列情况中, 这可能会非常实用:

    • 你必须针对不同的目的提供多个不同的构造函数, 但是其签名相同时。 例如, 在 Java、 C++、 C# 以及其他许多语言中不可能同时存在 Random­(int max) 和 Random­(int min) 函数。 最常用的变通方式是创建多个调用默认构造函数的静态方法, 并于稍后再设置适当的数值。
    • 你希望复用已有对象而不是初始化新对象时 (参考单例模式)。 绝大多数编程语言的构造函数必须都返回一个新的类实例。 静态构建方法是应对该限制的变通方法。 在静态方法内部, 你的代码会决定是调用构造函数创建一个全新实例, 还是返回一个在缓存中已有的对象。

    在下面的例子中, ​ load 方法是一个静态构建方法。 它提供了一种从数据库中获取用户的灵活方式。

    class User {
        private $id, $name, $email, $phone;
    
        public function __construct($id, $name, $email, $phone) {
            $this->id = $id;
            $this->name = $name;
            $this->email = $email;
            $this->phone = $phone;
        }
    
        public static function load($id) {
            list($id, $name, $email, $phone) = DB::load_data('users', 'id', 'name', 'email', 'phone');
            $user = new User($id, $name, $email, $phone);
            return $user;
        }
    }
    

    4. 简单工厂模式

    简单工厂模式 描述了一个类, 它拥有一个包含大量条件语句的构建方法, 可根据方法的参数来选择对何种产品进行初始化并将其返回。

    人们通常会将简单工厂与普通的工厂或其它创建型设计模式混淆。 在绝大多数情况下, 简单工厂是引入工厂方法或抽象工厂模式时的一个中间步骤。

    简单工厂通常没有子类。 但当从一个简单工厂中抽取出子类后, 它看上去就会更像经典的工厂方法模式了。

    顺便提一句, 如果你将一个简单工厂声明为 abstract 类型, 它并不会神奇地变成抽象工厂模式。

    这里是一个简单工厂的例子:

    class UserFactory {
        public static function create($type) {
            switch ($type) {
                case 'user': return new User();
                case 'customer': return new Customer();
                case 'admin': return new Admin();
                default:
                    throw new Exception('传递的用户类型错误。');
            }
        }
    }
    

    5. 工厂方法模式

    工厂方法 是一种创建型设计模式, 其在父类中提供一个创建对象的方法, 允许子类决定实例化对象的类型。

    如果在基类及其扩展的子类中都有一个构建方法的话, 那它可能就是工厂方法。

    abstract class Department {
        public abstract function createEmployee($id);
    
        public function fire($id) {
            $employee = $this->createEmployee($id);
            $employee->paySalary();
            $employee->dismiss();
        }
    }
    
    class ITDepartment extends Department {
        public function createEmployee($id) {
            return new Programmer($id);
        }
    }
    
    class AccountingDepartment extends Department {
        public function createEmployee($id) {
            return new Accountant($id);
        }
    }
    

    6. 抽象工厂模式

    抽象工厂 》 是一种创建型设计模式, 它能创建一系列相关或相互依赖的对象, 而无需指定其具体类。

    什么是 “系列对象”? 例如有这样一组的对象: ​ 运输工具 + 引擎 + 控制器 。 它可能会有几个变体:

    • 汽车 + 内燃机 + 方向盘
    • 飞机 + 喷气式发动机 + 操纵杆

    如果你的程序中并不涉及产品系列的话, 那就不需要抽象工厂。

    再次重申, 许多人分不清抽象工厂模式和声明为 abstract 的简单工厂。 不要犯这个错误!


    责任链模式、 命令模式、 中介者模式和观察者模式用于处理请求发送者和接收者之间的不同连接方式:

    • 责任链按照顺序将请求动态传递给一系列的潜在接收者, 直至其中一名接收者对请求进行处理。
    • 命令在发送者和请求者之间建立单向连接。
    • 中介者清除了发送者和请求者之间的直接连接, 强制它们通过一个中介对象进行间接沟通。
    • 观察者允许接收者动态地订阅或取消接收请求。

    中介者和观察者之间的区别往往很难记住。 在大部分情况下, 你可以使用其中一种模式, 而有时可以同时使用。 让我们来看看如何做到这一点:

    中介者的主要目标是消除一系列系统组件之间的相互依赖。 这些组件将依赖于同一个中介者对象。 观察者的目标是在对象之间建立动态的单向连接, 使得部分对象可作为其他对象的附属发挥作用。

    有一种流行的中介者模式实现方式依赖于观察者。 中介者对象担当发布者的角色, 其他组件则作为订阅者, 可以订阅中介者的事件或取消订阅。 当中介者以这种方式实现时, 它可能看上去与观察者非常相似。

    当你感到疑惑时, 记住可以采用其他方式来实现中介者。 例如, 你可永久性地将所有组件链接到同一个中介者对象。 这种实现方式和观察者并不相同, 但这仍是一种中介者模式。

    假设有一个程序, 其所有的组件都变成了发布者, 它们之间可以相互建立动态连接。 这样程序中就没有中心化的中介者对象, 而只有一些分布式的观察者。

  • 相关阅读:
    Binary Tree
    (并查集..含小总结)A
    (超简单并查集)POJ2524 Ubiquitous Religions(8.4.4)
    OpenSees开发(一)windows 上编译opensees (使用vs2005)
    vector的push_back对于拷贝构造和赋值操作的调用
    一个指针的引用引发的血案
    MFC中由左键单击模拟左键双击引起的问题
    CLAPACK动态调用
    SVN部署(远程)客户端篇
    SVN部署(本地)
  • 原文地址:https://www.cnblogs.com/stringarray/p/12793282.html
Copyright © 2011-2022 走看看