设计模式反模式:避免滥用设计模式的10个常见误区

2024-12-27 14:39 更新

Hello,大家好,我是V哥。很多文章都在介绍设计模式怎么用,讲解设计模式的原理等等,设计模式的思想是编程中的精髓,用好了可以让代码结构利于维护和扩展,同时代码风格也更加优雅,V 哥也写过这样一篇文章,但很少有人从反模式的角度来讲一讲,过度滥用设计模式将给项目带来灾难。

设计模式反模式(Anti-Pattern)是指那些表面上看起来像是设计模式,但实际上会导致软件设计问题的做法。这些做法可能会导致代码难以维护、扩展或测试。

在实际开发中,设计模式的误用可能会导致软件设计的问题,下面 V 哥整理了10种常见的设计模式误用案例,来看一下:

  1. 滥用工厂模式
    • 误用:创建一个万能工厂类来生成所有类型的实例,导致工厂类变得庞大且难以维护。
    • 正确使用:应该为不同类型的对象创建专门的工厂类,遵循单一职责原则。
  2. 过度使用单例模式
    • 误用:在不应该使用单例模式的场景中使用它,例如在需要频繁创建和销毁对象的系统中。
    • 正确使用:单例模式适用于全局配置或共享资源,但应该谨慎使用以避免全局状态的问题。
  3. 错误使用装饰器模式
    • 误用:在不支持扩展的类上使用装饰器模式,或者装饰器类与原始类耦合太紧。
    • 正确使用:确保装饰器模式用于可扩展的对象,并且装饰器应该只添加额外的行为,不改变原有对象的行为。
  4. 不恰当的继承使用
    • 误用:通过继承来实现代码复用,而不是基于“是一个(is-a)”的关系。
    • 正确使用:应该使用组合而不是继承来复用代码,继承应该仅用于表示类型层次结构。
  5. 滥用观察者模式
    • 误用:在不需要观察者模式的场景中使用它,或者将过多的逻辑放入观察者中。
    • 正确使用:观察者模式应该用于实现发布-订阅机制,而不是作为通信的唯一手段。
  6. 错误使用命令模式
    • 误用:将简单的操作也封装成命令对象,导致系统复杂度增加。
    • 正确使用:命令模式适用于需要记录、排队、撤销或日志记录的操作。
  7. 不恰当的状态模式使用
    • 误用:将状态模式用于简单的状态变化,或者状态转换逻辑过于复杂。
    • 正确使用:状态模式应该用于对象状态复杂且状态变化频繁的场景。
  8. 错误使用代理模式
    • 误用:在不需要代理的场景中使用代理模式,例如直接访问远程服务而不是使用本地代理。
    • 正确使用:代理模式应该用于控制对对象的访问,提供额外的安全或控制逻辑。
  9. 滥用策略模式
    • 误用:将策略模式用于只有一个或很少几个策略的场景,或者策略类与上下文类耦合太紧。
    • 正确使用:策略模式应该用于在运行时需要切换算法或行为的场景。
  10. 不恰当的组合/聚合使用
    • 误用:错误地使用组合/聚合来表示整体与部分的关系,或者管理不当导致内存泄漏。
    • 正确使用:应该明确整体与部分的关系,并且正确管理对象的生命周期。

下面我们来一一介绍这10种滥用设计模式的场景案例和分析。

1. 滥用工厂模式

滥用工厂模式通常指的是创建一个过于庞大和复杂的工厂类,它试图创建和管理系统中所有不同类型的对象。这种做法违反了设计模式的意图,即应该保持代码的简洁性和可维护性。

滥用工厂模式的案例

假设我们有一个应用程序,其中包含多个不同类型的产品,每个产品都有其特定的创建逻辑。在滥用工厂模式的情况下,我们可能会创建一个“万能工厂”来处理所有产品的创建。

public class UniversalFactory {
    // 这是一个滥用工厂模式的例子,工厂类过于庞大和复杂
    public ProductA createProductA() {
        // 复杂的创建逻辑
        return new ProductA();
    }


    public ProductB createProductB() {
        // 复杂的创建逻辑
        return new ProductB();
    }


    public ProductC createProductC() {
        // 复杂的创建逻辑
        return new ProductC();
    }


    // ... 更多的产品创建方法


    public static class ProductA {
        // ProductA 的实现
    }


    public static class ProductB {
        // ProductB 的实现
    }


    public static class ProductC {
        // ProductC 的实现
    }


    // ... 更多的产品类
}

在上面的代码中,UniversalFactory 包含了创建所有产品的方法。随着应用程序的发展,这个工厂类可能会变得越来越庞大,难以维护。每次添加新产品时,都需要修改工厂类,这违反了开闭原则(对扩展开放,对修改封闭)。

具体说明

  1. 违反开闭原则:每次添加新产品时,都需要修改工厂类,这增加了维护成本。
  2. 职责不明确:工厂类承担了过多的职责,它不仅负责创建对象,还可能包含了与产品相关的逻辑。
  3. 难以测试:由于工厂类与产品类紧密耦合,单元测试变得更加困难。
  4. 代码复杂性增加:随着工厂类变得越来越庞大,理解和使用它也变得更加复杂。

正确使用工厂模式

正确的做法是为每个产品系列创建专门的工厂类,这样可以更好地组织代码,并且每个工厂类只关注其特定的产品。

public class ProductAFactory {
    public ProductA create() {
        // 创建 ProductA 的逻辑
        return new ProductA();
    }
}


public class ProductBFactory {
    public ProductB create() {
        // 创建 ProductB 的逻辑
        return new ProductB();
    }
}


// ... 为其他产品创建更多的工厂类

在这个例子中,每个工厂类只负责创建一种类型的产品,这样代码更加清晰,也更容易维护和扩展。如果需要添加新产品,只需添加一个新的工厂类,而不需要修改现有的工厂类。

2. 过度使用单例模式

过度使用单例模式的案例

单例模式确保一个类只有一个实例,并提供一个全局访问点。然而,在一些业务场景中,过度使用单例模式可能会导致问题,尤其是当单例对象的状态需要在多个用户或线程之间共享时。

业务场景

假设我们正在开发一个多用户的博客平台,每个用户都可以发布博客文章。我们有一个BlogPostManager类,负责管理博客文章的创建、编辑和删除。

错误使用单例模式

public class BlogPostManager {
    private static BlogPostManager instance;
    private List<BlogPost> blogPosts;


    private BlogPostManager() {
        blogPosts = new ArrayList<>();
    }


    public static synchronized BlogPostManager getInstance() {
        if (instance == null) {
            instance = new BlogPostManager();
        }
        return instance;
    }


    public void addBlogPost(BlogPost blogPost) {
        blogPosts.add(blogPost);
    }


    public List<BlogPost> getBlogPosts() {
        return blogPosts;
    }


    // ... 其他管理博客文章的方法
}


public class BlogPost {
    private String title;
    private String content;
    // ... 其他博客文章属性和方法
}

在这个例子中,BlogPostManager被设计为单例,这意味着所有用户共享同一个BlogPostManager实例和它管理的博客文章列表。这在多用户环境中是不安全的,因为一个用户的操作可能会影响其他用户看到的数据。

具体说明

  1. 共享状态问题:单例模式导致的全局状态使得在多用户环境中难以维护独立用户的数据隔离。
  2. 线程安全问题:在多线程环境中,单例模式可能会导致线程安全问题,尤其是在懒汉式初始化的情况下。
  3. 测试困难:单例模式使得依赖注入变得困难,因为它通常依赖于全局状态,这会影响单元测试的编写。
  4. 扩展性问题:当系统需要扩展时,单例模式可能会导致扩展性问题,因为它限制了对象的实例化方式。

正确使用单例模式

正确的做法是确保每个用户都有自己的BlogPostManager实例,或者使用依赖注入来管理BlogPostManager的生命周期。

使用依赖注入

public class BlogPostManager {
    private List<BlogPost> blogPosts;


    public BlogPostManager() {
        blogPosts = new ArrayList<>();
    }


    public void addBlogPost(BlogPost blogPost) {
        blogPosts.add(blogPost);
    }


    public List<BlogPost> getBlogPosts() {
        return blogPosts;
    }


    // ... 其他管理博客文章的方法
}


// 在用户类中
public class User {
    private BlogPostManager blogPostManager;


    public User(BlogPostManager blogPostManager) {
        this.blogPostManager = blogPostManager;
    }


    // ... 用户的方法
}

在这个例子中,每个User对象都有一个自己的BlogPostManager实例,这样可以确保用户之间的数据隔离。这种方式更适合多用户环境,并且使得单元测试更加容易。

所以啊,单例模式应该谨慎使用,特别是在需要数据隔离的多用户系统中。在这些情况下,考虑使用依赖注入或其他设计模式来替代单例模式。

3. 错误使用装饰器模式

错误使用装饰器模式的案例

装饰器模式是一种结构型设计模式,它允许向一个现有的对象添加新的功能,同时又不改变其结构。这种设计模式通过创建一个包装对象,即装饰器,来封装实际对象。

业务场景

假设我们有一个在线商店,其中有一个Product类,表示商店中的商品。我们希望通过装饰器模式为商品添加不同的包装选项,如礼品包装。

错误使用装饰器模式

public interface Product {
    double getCost();
    String getDescription();
}


public class Book implements Product {
    private String title;


    public Book(String title) {
        this.title = title;
    }


    @Override
    public double getCost() {
        return 15.00;
    }


    @Override
    public String getDescription() {
        return "Book: " + title;
    }
}


public abstract class ProductDecorator implements Product {
    protected Product decoratedProduct;


    public ProductDecorator(Product decoratedProduct) {
        this.decoratedProduct = decoratedProduct;
    }


    public double getCost() {
        return decoratedProduct.getCost();
    }


    public String getDescription() {
        return decoratedProduct.getDescription();
    }
}


public class GiftWrapDecorator extends ProductDecorator {
    private double giftWrapCost = 3.00;


    public GiftWrapDecorator(Product decoratedProduct) {
        super(decoratedProduct);
    }


    @Override
    public double getCost() {
        return decoratedProduct.getCost() + giftWrapCost;
    }


    @Override
    public String getDescription() {
        return decoratedProduct.getDescription() + ", with gift wrap";
    }
}


// 错误使用装饰器模式的客户端代码
public class Main {
    public static void main(String[] args) {
        Product book = new Book("Java Design Patterns");
        Product giftWrappedBook = new GiftWrapDecorator(book);


        System.out.println("Cost: " + giftWrappedBook.getCost()); // Cost: 18.0
        System.out.println("Description: " + giftWrappedBook.getDescription()); // Description: Book: Java Design Patterns, with gift wrap
    }
}

在这个例子中,GiftWrapDecorator 装饰器正确地为Book 添加了礼品包装功能。然而,如果我们错误地将装饰器模式应用于不应该被装饰的对象,或者在装饰器中引入了过多的逻辑,就可能导致问题。

具体说明

  1. 过度装饰:如果一个对象被多层装饰,可能会导致对象的复杂度过高,难以理解和维护。
  2. 装饰器逻辑过于复杂:装饰器应该只负责添加额外的功能,如果装饰器内部逻辑过于复杂,可能会使得整个系统难以维护。
  3. 违反开闭原则:如果每次需要添加新功能时都需要修改装饰器或引入新的装饰器,这可能违反了开闭原则,即对扩展开放,对修改封闭。
  4. 性能问题:装饰器模式可能会导致性能问题,因为每次装饰都可能增加额外的开销。

正确使用装饰器模式

正确的做法是确保装饰器只添加必要的功能,并且装饰器的逻辑应该尽量简单。

// 正确的客户端代码
public class Main {
    public static void main(String[] args) {
        Product book = new Book("Java Design Patterns");
        Product giftWrappedBook = new GiftWrapDecorator(book);


        System.out.println("Cost: " + giftWrappedBook.getCost()); // Cost: 18.0
        System.out.println("Description: " + giftWrappedBook.getDescription()); // Description: Book: Java Design Patterns, with gift wrap
    }
}

修改后的例子中,我们只添加了必要的装饰器。如果需要更多的装饰功能,我们应该引入新的装饰器类,而不是在现有装饰器中添加更多的逻辑。

因此我们人认识到,装饰器模式是一种强大的设计模式,但应该谨慎使用,确保它不会使系统变得过于复杂或难以维护。

4. 不恰当的继承使用

不恰当的继承使用案例

继承是一种强大的工具,但它应该谨慎使用。不恰当的继承通常是由于错误地将“是一个(is-a)”关系应用于代码中,而不是基于功能的共享。

业务场景

假设我们有一个电子商务平台,需要处理不同类型的支付方式。我们可能会错误地使用继承来实现这些支付方式。

不恰当的继承使用

// 基类 PaymentMethod 错误地被用作所有支付方式的父类
public abstract class PaymentMethod {
    protected String name;


    public PaymentMethod(String name) {
        this.name = name;
    }


    public abstract boolean processPayment(double amount);
}


// 信用卡支付类错误地继承了 PaymentMethod
public class CreditCardPayment extends PaymentMethod {
    private String cardNumber;
    private String cardHolderName;
    private String expirationDate;
    private String cvv;


    public CreditCardPayment(String name, String cardNumber, String cardHolderName, String expirationDate, String cvv) {
        super(name);
        this.cardNumber = cardNumber;
        this.cardHolderName = cardHolderName;
        this.expirationDate = expirationDate;
        this.cvv = cvv;
    }


    @Override
    public boolean processPayment(double amount) {
        // 处理信用卡支付逻辑
        return true;
    }
}


// 银行转账支付类错误地继承了 PaymentMethod
public class BankTransferPayment extends PaymentMethod {
    private String accountNumber;
    private String bankName;


    public BankTransferPayment(String name, String accountNumber, String bankName) {
        super(name);
        this.accountNumber = accountNumber;
        this.bankName = bankName;
    }


    @Override
    public boolean processPayment(double amount) {
        // 处理银行转账支付逻辑
        return true;
    }
}

在这个例子中,PaymentMethod 被用作所有支付方式的基类。然而,这并不是一个恰当的使用继承的情况,因为“信用卡支付”和“银行转账支付”并不是“支付方式”的一种类型,它们是具体的支付实现。这种继承关系违反了“是一个(is-a)”原则,因为信用卡支付并不是支付方式的一种类型,而是一种具体的支付行为。

具体说明

  1. 违反“是一个(is-a)”原则:继承应该表示一个类是另一个类的特化。如果一个类不是另一个类的特化,那么继承就是不恰当的。
  2. 增加耦合性:继承自同一个基类的所有子类都与基类紧密耦合。如果基类发生变化,所有的子类都可能受到影响。
  3. 限制了灵活性:继承自基类的子类通常不能轻易地切换到另一种类型的基类,这限制了设计的灵活性。
  4. 继承导致的复杂性:继承层次结构可能会变得复杂和难以理解,特别是当它们很深或很宽时。

正确的使用继承

正确的做法是使用组合而不是继承来实现支付方式。

// 支付方式接口
public interface PaymentMethod {
    boolean processPayment(double amount);
}


// 信用卡支付类实现支付方式接口
public class CreditCardPayment implements PaymentMethod {
    private String cardNumber;
    private String cardHolderName;
    private String expirationDate;
    private String cvv;


    public CreditCardPayment(String cardNumber, String cardHolderName, String expirationDate, String cvv) {
        this.cardNumber = cardNumber;
        this.cardHolderName = cardHolderName;
        this.expirationDate = expirationDate;
        this.cvv = cvv;
    }


    @Override
    public boolean processPayment(double amount) {
        // 处理信用卡支付逻辑
        return true;
    }
}


// 银行转账支付类实现支付方式接口
public class BankTransferPayment implements PaymentMethod {
    private String accountNumber;
    private String bankName;


    public BankTransferPayment(String accountNumber, String bankName) {
        this.accountNumber = accountNumber;
        this.bankName = bankName;
    }


    @Override
    public boolean processPayment(double amount) {
        // 处理银行转账支付逻辑
        return true;
    }
}

改进后,我们使用接口PaymentMethod来定义支付行为,而不是使用继承。这样,不同的支付方式可以自由地实现这个接口,而不需要从特定的基类继承。这种设计更加灵活,每个支付方式的具体实现都是独立的,不会受到其他实现的影响。

5. 滥用观察者模式

滥用观察者模式的案例

观察者模式是一种行为设计模式,它定义了对象之间的一对多依赖关系,当一个对象状态改变时,所有依赖于它的对象都会得到通知。滥用观察者模式可能会导致性能问题、代码复杂性增加以及难以维护的代码。

业务场景

假设我们有一个新闻发布平台,每当有新的新闻发布时,所有订阅者都应该收到通知。如果系统中有大量的订阅者或者通知逻辑非常复杂,滥用观察者模式可能会导致问题。

滥用观察者模式的代码示例

import java.util.ArrayList;
import java.util.List;


// 主题接口
public interface Observer {
    void update(String news);
}


// 具体主题
public class NewsPublisher {
    private List<Observer> observers;
    private String news;


    public NewsPublisher() {
        observers = new ArrayList<>();
    }


    public void addObserver(Observer observer) {
        observers.add(observer);
    }


    public void removeObserver(Observer observer) {
        observers.remove(observer);
    }


    public void notifyObservers() {
        for (Observer observer : observers) {
            observer.update(news);
        }
    }


    public void setNews(String news) {
        this.news = news;
        notifyObservers();
    }


    public String getNews() {
        return news;
    }
}


// 具体观察者
public class NewsSubscriber implements Observer {
    private String name;


    public NewsSubscriber(String name) {
        this.name = name;
    }


    @Override
    public void update(String news) {
        System.out.println(name + " received news: " + news);
    }
}


// 客户端代码
public class NewsPlatform {
    public static void main(String[] args) {
        NewsPublisher publisher = new NewsPublisher();
        publisher.addObserver(new NewsSubscriber("Subscriber 1"));
        publisher.addObserver(new NewsSubscriber("Subscriber 2"));
        // ... 添加更多订阅者


        // 发布新闻
        publisher.setNews("Breaking news!");
    }
}

在这个例子中,NewsPublisher 是一个具体主题,它维护了一个观察者列表,并在新闻更新时通知所有观察者。NewsSubscriber 是一个具体观察者,它实现了Observer接口,并在接收到新闻时打印出来。

具体说明

  1. 性能问题:如果观察者数量非常多,每次状态变化时通知所有观察者可能会导致性能问题。
  2. 代码复杂性:在观察者模式中添加复杂的业务逻辑可能会导致代码难以理解和维护。
  3. 循环依赖:如果观察者和主题之间存在循环依赖,可能会导致难以追踪的错误。
  4. 内存泄漏:如果观察者没有正确地从主题中移除,可能会导致内存泄漏。

正确的使用观察者模式

正确的做法是确保观察者模式的使用场景适合,并且观察者的数量和通知逻辑都得到了合理控制。

// 客户端代码
public class NewsPlatform {
    public static void main(String[] args) {
        NewsPublisher publisher = new NewsPublisher();
        publisher.addObserver(new NewsSubscriber("Subscriber 1"));
        // 添加适量的订阅者,而不是无限制地添加


        // 发布新闻
        publisher.setNews("Breaking news!");
    }
}

在这个修正后的客户端代码中,我们只添加了适量的订阅者。此外,我们应该确保在不再需要接收通知时,从主题中移除观察者,以避免内存泄漏。

最后强调一下,观察者模式是一种有用的设计模式,但应该在适当的场景中使用,并且要注意控制观察者的数量和复杂性。在设计系统时,应该考虑到性能和可维护性。

6. 错误使用命令模式

错误使用命令模式的案例

命令模式是一种行为设计模式,它将请求封装为一个对象,从而允许用户使用不同的请求、队列或日志请求来参数化其他对象。命令模式也支持可撤销的操作。错误使用命令模式可能会导致系统复杂性增加、代码冗余或者违反开闭原则。

业务场景

假设我们有一个简单的文本编辑器,用户可以对文本执行一些操作,如插入文本、删除文本等。如果我们错误地将所有操作都封装为命令对象,即使这些操作可以通过更简单的方法实现,这就是错误使用命令模式。

错误使用命令模式的代码示例

// 命令接口
public interface Command {
    void execute();
}


// 简单的文本编辑器类
public class TextEditor {
    private String content = "";


    public void type(String text) {
        content += text;
    }


    public void removeLastWord() {
        content = content.replaceAll("\\s+\\S+$", "");
    }


    public String getContent() {
        return content;
    }
}


// 插入文本命令类
public class TypeCommand implements Command {
    private TextEditor editor;
    private String text;


    public TypeCommand(TextEditor editor, String text) {
        this.editor = editor;
        this.text = text;
    }


    @Override
    public void execute() {
        editor.type(text);
    }
}


// 删除文本命令类
public class RemoveCommand implements Command {
    private TextEditor editor;


    public RemoveCommand(TextEditor editor) {
        this.editor = editor;
    }


    @Override
    public void execute() {
        editor.removeLastWord();
    }
}


// 客户端代码
public class EditorClient {
    public static void main(String[] args) {
        TextEditor editor = new TextEditor();
        Command typeCommand = new TypeCommand(editor, "Hello World");
        Command removeCommand = new RemoveCommand(editor);


        typeCommand.execute();
        System.out.println(editor.getContent()); // 输出: Hello World


        removeCommand.execute();
        System.out.println(editor.getContent()); // 输出: (empty string)
    }
}

在这个例子中,我们为TextEditor的每个操作都创建了一个命令对象。然而,对于这样简单的操作,使用命令模式可能是过度设计。这不仅增加了系统的复杂性,而且也增加了代码的冗余。

具体说明

  1. 过度设计:对于简单的操作,使用命令模式可能会引入不必要的复杂性。
  2. 违反开闭原则:如果新操作需要添加,我们不得不为每个新操作创建新的命令类,这违反了对扩展开放,对修改封闭的原则。
  3. 增加学习成本:新开发人员可能需要花费额外的时间去理解命令模式的使用,而不是直接使用简单的方法调用。
  4. 性能考虑:对于性能敏感的应用,命令对象的创建和管理可能会带来额外的性能开销。

正确的使用命令模式

正确的做法是将命令模式用于确实需要它的场合,比如操作的撤销/重做功能、操作的排队执行等。

// 撤销命令接口
public interface Command {
    void execute();
    void undo();
}


// 插入文本命令类
public class TypeCommand implements Command {
    private TextEditor editor;
    private String text;
    private String previousContent;


    public TypeCommand(TextEditor editor, String text) {
        this.editor = editor;
        this.text = text;
    }


    @Override
    public void execute() {
        previousContent = editor.getContent();
        editor.type(text);
    }


    @Override
    public void undo() {
        editor.setContent(previousContent);
    }
}


// 客户端代码
public class EditorClient {
    public static void main(String[] args) {
        TextEditor editor = new TextEditor();
        Command typeCommand = new TypeCommand(editor, "Hello World");


        typeCommand.execute();
        System.out.println(editor.getContent()); // 输出: Hello World


        typeCommand.undo();
        System.out.println(editor.getContent()); // 输出: (empty string)
    }
}

在这个修正后的示例中,我们为需要撤销功能的命令实现了undo方法。这样,命令模式的使用就变得更加合理,因为它提供了额外的价值,即操作的撤销功能。

7. 不恰当的状态模式使用

不恰当的状态模式使用案例

状态模式是一种行为设计模式,它允许一个对象在其内部状态改变时改变其行为。这种模式非常适合于那些具有多个状态,并且在不同状态下行为有显著差异的对象。不恰当的使用状态模式可能会导致设计复杂、难以维护和理解。

业务场景

假设我们有一个简单的交通信号灯系统,它有三个状态:红灯、绿灯和黄灯。每个状态持续一定时间后会转换到下一个状态。如果我们错误地使用状态模式来处理这个简单的顺序逻辑,就可能导致不恰当的设计。

不恰当的状态模式使用代码示例

// 状态接口
public interface TrafficLightState {
    void change(TrafficLight light);
}


// 红灯状态类
public class RedLight implements TrafficLightState {
    @Override
    public void change(TrafficLight light) {
        System.out.println("Red light on");
        // 假设红灯持续一段时间后自动切换到绿灯
        light.setState(new GreenLight());
    }
}


// 绿灯状态类
public class GreenLight implements TrafficLightState {
    @Override
    public void change(TrafficLight light) {
        System.out.println("Green light on");
        // 假设绿灯持续一段时间后自动切换到黄灯
        light.setState(new YellowLight());
    }
}


// 黄灯状态类
public class YellowLight implements TrafficLightState {
    @Override
    public void change(TrafficLight light) {
        System.out.println("Yellow light on");
        // 假设黄灯持续一段时间后自动切换到红灯
        light.setState(new RedLight());
    }
}


// 交通信号灯类
public class TrafficLight {
    private TrafficLightState state;


    public TrafficLight() {
        this.state = new RedLight(); // 初始状态为红灯
    }


    public void setState(TrafficLightState state) {
        this.state = state;
    }


    public void change() {
        state.change(this);
    }
}


// 客户端代码
public class TrafficLightSystem {
    public static void main(String[] args) {
        TrafficLight light = new TrafficLight();

        
        // 模拟信号灯变化
        light.change(); // 红灯
        light.change(); // 绿灯
        light.change(); // 黄灯
        light.change(); // 再次红灯
    }
}

在这个例子中,我们为交通信号灯的每个状态都创建了一个状态类。然而,对于这样一个简单的顺序逻辑,使用状态模式可能是过度设计。这不仅增加了系统的复杂性,而且也增加了代码的冗余。

具体说明

  1. 过度设计:对于简单的状态转换逻辑,使用状态模式可能会引入不必要的复杂性。
  2. 增加学习成本:新开发人员可能需要花费额外的时间去理解状态模式的使用,而不是直接使用简单的状态管理逻辑。
  3. 代码冗余:每个状态类都需要实现相同的change方法,这可能导致代码冗余。
  4. 难以维护:随着系统的发展,维护和扩展状态模式可能会变得复杂,特别是当状态和转换逻辑变得更加复杂时。

正确的使用状态模式

正确的做法是将状态模式用于确实需要它的场合,比如状态转换逻辑复杂、状态之间有显著不同的行为或者需要记录状态历史等。

// 交通信号灯类(简化版)
public class TrafficLight {
    private String[] lights = {"Red", "Green", "Yellow"};
    private int index = 0; // 初始状态为红灯


    public void change() {
        System.out.println(lights[index] + " light on");
        index = (index + 1) % lights.length; // 循环切换状态
    }
}


// 客户端代码
public class TrafficLightSystem {
    public static void main(String[] args) {
        TrafficLight light = new TrafficLight();

        
        // 模拟信号灯变化
        light.change(); // 红灯
        light.change(); // 绿灯
        light.change(); // 黄灯
        light.change(); // 再次红灯
    }
}

在这个修正后的示例中,我们使用一个简单的索引和数组来管理信号灯的状态,而不是使用状态模式。这种设计更加简洁和直观,适合处理简单的状态转换逻辑。

8. 错误使用代理模式

错误使用代理模式的案例

代理模式是一种结构型设计模式,它提供了对另一个对象的代理以控制对这个对象的访问。代理模式有多种类型,包括虚拟代理、远程代理、保护代理和智能引用代理等。错误使用代理模式可能会导致系统设计过于复杂、性能降低或者违反设计原则。

业务场景

假设我们有一个图像加载系统,需要从网络加载图像。如果我们错误地为每个图像对象创建一个代理,即使这些图像对象不需要代理提供的额外功能,这就是错误使用代理模式。

错误使用代理模式的代码示例

// 目标接口
public interface Image {
    void display();
}


// 目标类
public class RealImage implements Image {
    private String fileName;


    public RealImage(String fileName) {
        this.fileName = fileName;
        loadFromDisk(fileName);
    }


    @Override
    public void display() {
        System.out.println("Displaying " + fileName);
    }


    private void loadFromDisk(String fileName) {
        System.out.println("Loading " + fileName + " from disk");
    }
}


// 代理类
public class ImageProxy implements Image {
    private RealImage realImage;
    private String fileName;


    public ImageProxy(String fileName) {
        this.fileName = fileName;
    }


    @Override
    public void display() {
        if (realImage == null) {
            realImage = new RealImage(fileName);
        }
        realImage.display();
    }
}


// 客户端代码
public class ProxyExample {
    public static void main(String[] args) {
        Image image = new ImageProxy("image1.jpg");
        image.display();
    }
}

在这个例子中,ImageProxy 代理类包装了 RealImage 目标类。然而,如果 RealImage 类的实例化和使用非常简单,并且不需要延迟加载或其他代理功能,那么使用代理模式就是不恰当的。这可能会导致不必要的性能开销和设计复杂性。

具体说明

  1. 不必要的复杂性:如果目标对象不需要代理提供的控制或延迟加载等功能,使用代理模式会增加系统的复杂性。
  2. 性能开销:代理对象的创建和使用可能会引入额外的性能开销,尤其是在不需要代理的情况下。
  3. 违反开闭原则:如果未来需要添加新的代理功能,可能需要修改代理类,这违反了对扩展开放,对修改封闭的原则。
  4. 增加维护成本:代理模式可能会导致代码难以理解和维护,尤其是当代理类和目标类之间的关系不明确时。

正确的使用代理模式

正确的做法是将代理模式用于确实需要它的场合,比如需要控制对资源的访问、延迟加载资源或者提供额外的安全控制等。

// 虚拟代理类
public class VirtualImageProxy implements Image {
    private RealImage realImage;
    private String fileName;


    public VirtualImageProxy(String fileName) {
        this.fileName = fileName;
    }


    @Override
    public void display() {
        if (realImage == null) {
            realImage = new RealImage(fileName);
        }
        realImage.display();
    }


    public void loadImage() {
        System.out.println("Loading " + fileName + " from network");
    }
}


// 客户端代码
public class ProxyExample {
    public static void main(String[] args) {
        Image image = new VirtualImageProxy("image1.jpg");
        image.loadImage(); // 模拟从网络加载图像
        image.display();
    }
}

在这个修正后的示例中,VirtualImageProxy 类提供了延迟加载的功能,只有在实际需要显示图像时才从网络加载。这种设计更加合理,因为它提供了代理模式的实际价值,即延迟加载和优化性能。

9. 滥用策略模式

滥用策略模式的案例

策略模式是一种行为设计模式,它定义了一系列的算法,并将每一个算法封装起来,使它们可以互换使用。策略模式的用意是使算法的变化不会影响到使用算法的用户。滥用策略模式可能会导致设计过于复杂,增加不必要的类数量,以及使得代码难以理解和维护。

业务场景

假设我们有一个简单的计算器程序,它可以执行加、减、乘、除四种基本运算。如果我们为每一种操作都创建一个策略类,即使这些操作可以通过更简单的方法实现,这就是滥用策略模式。

滥用策略模式的代码示例

// 策略接口
public interface CalculationStrategy {
    int calculate(int a, int b);
}


// 加法策略类
public class AddStrategy implements CalculationStrategy {
    @Override
    public int calculate(int a, int b) {
        return a + b;
    }
}


// 减法策略类
public class SubtractStrategy implements CalculationStrategy {
    @Override
    public int calculate(int a, int b) {
        return a - b;
    }
}


// 乘法策略类
public class MultiplyStrategy implements CalculationStrategy {
    @Override
    public int calculate(int a, int b) {
        return a * b;
    }
}


// 除法策略类
public class DivideStrategy implements CalculationStrategy {
    @Override
    public int calculate(int a, int b) {
        if (b != 0) {
            return a / b;
        } else {
            throw new IllegalArgumentException("Divider cannot be zero.");
        }
    }
}


// 计算器上下文
public class Calculator {
    private CalculationStrategy strategy;


    public Calculator(CalculationStrategy strategy) {
        this.strategy = strategy;
    }


    public void setStrategy(CalculationStrategy strategy) {
        this.strategy = strategy;
    }


    public int performCalculation(int a, int b) {
        return strategy.calculate(a, b);
    }
}


// 客户端代码
public class StrategyPatternExample {
    public static void main(String[] args) {
        Calculator calculator = new Calculator(new AddStrategy());
        System.out.println("10 + 5 = " + calculator.performCalculation(10, 5));


        calculator.setStrategy(new SubtractStrategy());
        System.out.println("10 - 5 = " + calculator.performCalculation(10, 5));


        calculator.setStrategy(new MultiplyStrategy());
        System.out.println("10 * 5 = " + calculator.performCalculation(10, 5));


        calculator.setStrategy(new DivideStrategy());
        System.out.println("10 / 5 = " + calculator.performCalculation(10, 5));
    }
}

在这个例子中,我们为计算器的每一种操作都创建了一个策略类。然而,对于这样简单的操作,使用策略模式可能是过度设计。这不仅增加了系统的复杂性,而且也增加了代码的冗余。

具体说明

  1. 过度设计:对于简单的操作,使用策略模式可能会引入不必要的复杂性。
  2. 增加学习成本:新开发人员可能需要花费额外的时间去理解策略模式的使用,而不是直接使用简单的方法调用。
  3. 代码冗余:每个策略类都需要实现相同的接口,这可能导致代码冗余。
  4. 难以维护:随着系统的发展,维护和扩展策略模式可能会变得复杂,特别是当策略类的数量增加时。

正确的使用策略模式

正确的做法是将策略模式用于确实需要它的场合,比如算法的选择会频繁变化,或者需要在运行时根据不同的条件选择不同的算法。

// 计算器类(简化版)
public class SimpleCalculator {
    public int add(int a, int b) {
        return a + b;
    }


    public int subtract(int a, int b) {
        return a - b;
    }


    public int multiply(int a, int b) {
        return a * b;
    }


    public int divide(int a, int b) {
        if (b != 0) {
            return a / b;
        } else {
            throw new IllegalArgumentException("Divider cannot be zero.");
        }
    }
}


// 客户端代码
public class SimpleCalculatorExample {
    public static void main(String[] args) {
        SimpleCalculator calculator = new SimpleCalculator();
        System.out.println("10 + 5 = " + calculator.add(10, 5));
        System.out.println("10 - 5 = " + calculator.subtract(10, 5));
        System.out.println("10 * 5 = " + calculator.multiply(10, 5));
        System.out.println("10 / 5 = " + calculator.divide(10, 5));
    }
}

修正后的代码,我们直接在计算器类中实现了所有的操作,而不是使用策略模式。这种设计更加简洁和直观,适合处理简单的操作。如果未来需要添加新的算法或者改变算法的行为,可以考虑使用策略模式来提高灵活性。

10. 不恰当的组合/聚合使用

不恰当的组合/聚合使用案例

组合(Composition)和聚合(Aggregation)是表示整体与部分关系的方式,它们都是关联的特殊形式。不恰当地使用组合或聚合可能会导致对象的生命周期管理混乱、职责不明确或者设计过于复杂。

业务场景

假设我们有一个公司管理系统,其中包含公司和员工的关系。如果错误地使用组合或聚合,可能会导致管理混乱,比如错误地删除公司时也删除了员工。

不恰当的组合/聚合使用代码示例

// 员工类
public class Employee {
    private String name;
    private Company company; // 员工所属的公司


    public Employee(String name, Company company) {
        this.name = name;
        this.company = company;
    }


    // ... 员工的其他方法
}


// 公司类
public class Company {
    private String name;
    private List<Employee> employees; // 公司的员工列表


    public Company(String name) {
        this.name = name;
        this.employees = new ArrayList<>();
    }


    public void addEmployee(Employee employee) {
        employees.add(employee);
        employee.setCompany(this); // 错误地在添加员工时设置公司
    }


    public void removeEmployee(Employee employee) {
        employees.remove(employee);
        employee.setCompany(null); // 错误地在删除员工时取消设置公司
    }


    // ... 公司的其他方法
}


// 客户端代码
public class CompanyManagementSystem {
    public static void main(String[] args) {
        Company company = new Company("Moonshot AI");
        Employee employee1 = new Employee("Kimi", company);
        Employee employee2 = new Employee("Alex", company);


        company.addEmployee(employee1);
        company.addEmployee(employee2);


        // 错误地删除公司,同时删除了所有员工
        company = null; // 这将导致员工对象也被垃圾回收
    }
}

在这个例子中,Company 类通过 employees 列表与 Employee 类建立了一种关系。然而,当公司被设置为 null 时,由于员工对象仍然持有对公司的引用,这可能会导致不一致的状态,因为员工对象仍然认为它们属于一个已经不存在的公司。

具体说明

  1. 生命周期管理混乱:如果组合或聚合的对象生命周期没有正确管理,可能会导致对象状态不一致。
  2. 职责不明确:在不恰当的组合/聚合关系中,对象的职责可能不明确,比如员工对象应该只关心自己的数据,而不是关心所属公司的状态。
  3. 设计过于复杂:不恰当的关系可能会导致设计过于复杂,难以理解和维护。

正确的使用组合/聚合

正确的做法是确保组合/聚合关系反映了实际的业务逻辑,并且对象的生命周期得到了正确的管理。

// 员工类
public class Employee {
    private String name;
    private Company company; // 员工所属的公司


    public Employee(String name) {
        this.name = name;
    }


    // ... 员工的其他方法


    public void setCompany(Company company) {
        this.company = company;
    }


    public Company getCompany() {
        return company;
    }
}


// 公司类
public class Company {
    private String name;
    private List<Employee> employees; // 公司的员工列表


    public Company(String name) {
        this.name = name;
        this.employees = new ArrayList<>();
    }


    public void addEmployee(Employee employee) {
        employees.add(employee);
        employee.setCompany(this); // 正确地在添加员工时设置公司
    }


    public void removeEmployee(Employee employee) {
        employees.remove(employee);
        // 不取消设置公司,因为员工可能仍然需要知道他们之前属于哪个公司
    }


    // ... 公司的其他方法
}


// 客户端代码
public class CompanyManagementSystem {
    public static void main(String[] args) {
        Company company = new Company("Moonshot AI");
        List<Employee> employees = new ArrayList<>();
        employees.add(new Employee("Kimi"));
        employees.add(new Employee("Alex"));


        for (Employee employee : employees) {
            company.addEmployee(employee);
        }


        // 删除公司时,员工列表被清空,但员工对象仍然存在
        company = null;
        // 员工对象不会被垃圾回收,因为它们在 employees 列表中仍然有引用
    }
}

修正后的代码,我们确保了员工对象的生命周期独立于公司对象。即使公司对象被删除,员工对象仍然存在,并且可以通过其他方式进行管理。这种设计更加符合实际的业务逻辑,并且使得对象的生命周期管理更加清晰。

最后

滥用设计模式,就像在没有正确诊断的情况下给病人开药一样,可能会带来一系列的问题和后果。

  1. 增加复杂性: 设计模式应该用来简化设计,但如果滥用,它们会使系统变得更加复杂和难以理解。这就像给一个简单的房子增加了不必要的装饰,最终导致维护成本上升。
  2. 性能问题: 一些设计模式,如代理模式或装饰器模式,可能会引入额外的间接层,这可能会影响系统的性能。就像在赛车上增加不必要的重量,会降低它的速度。
  3. 代码膨胀: 滥用设计模式可能导致项目中存在大量不必要的类和接口,从而导致代码膨胀。这就像是在图书馆里增加了过多的书架,而书籍却寥寥无几。
  4. 维护困难: 当设计模式被不恰当地应用时,新来的开发者可能需要花费更多的时间来理解和维护代码。这就像是在没有地图的情况下探索一个迷宫,既费时又费力。
  5. 违反开闭原则: 设计模式应该帮助我们构建可扩展的系统,但如果滥用,每次需要改变时都可能需要修改现有代码,这违反了开闭原则。这就像是每次需要增加房间时,都必须拆除现有房屋的一部分。
  6. 过度耦合: 不恰当的设计模式使用可能导致类和对象之间的过度耦合,使得代码难以重用和测试。这就像是把所有的家具都固定在房间里,无法移动或重新布置。
  7. 资源浪费: 滥用设计模式可能会导致资源的浪费,因为开发者可能花费大量时间在不必要的设计上,而不是解决实际问题。这就像是在不需要空调的地方安装了昂贵的空调系统。
  8. 项目延期: 由于上述所有问题,滥用设计模式可能会导致项目延期。这就像是在建造桥梁时不断改变设计,导致工程无法按时完成。

设计模式是工具箱中的工具,它们应该根据项目的具体需求谨慎选择和使用。正确的设计模式应用可以提高代码质量、可维护性和可扩展性,而滥用则可能导致项目陷入混乱和灾难。所以咱们应该深入理解每种设计模式的适用场景,并在实际开发中做出明智的选择。原创不易,如果文章对你有帮助,欢迎点赞关注威哥爱编程,学习路上我们不孤单。

以上内容是否对您有帮助:
在线笔记
App下载
App下载

扫描二维码

下载编程狮App

公众号
微信公众号

编程狮公众号