设计模式之状态模式

在日常开发中,我们经常会遇到这样的场景:一个对象在不同时刻有不同的状态,不同状态下它的行为也会发生变化。此时,使用大量if...else或switch语句会让代码变得混乱而难以维护。为了更优雅地应对这种问题,状态模式(State Pattern)应运而生。

1. 概念

状态模式是行为型模式之一。它允许一个对象在其内部状态发生变化,改变它的行为,看起来就像修改了它的类一样。

简单来说:对象的行为取决于它的状态,并且可以在运行时根据状态改变行为。

状态模式主要包含以下角色:

**环境类(Context):**也称为上下文,它定义了客户端需要的接口,内部维护一个当前状态,并负责具体状态的切换。

**抽象状态类(State):**定义一个接口,用以封装环境对象中的特定状态所对应的行为,可以有一个或多个行为。

**具体状态类(Concrete State):**实现抽象状态所对应的行为,并且在需要的情况下进行状态切换。

2. 代码实现

这里我们以一个简单的订单状态来举例,我们一个订单可能拥有很多状态,比如代付款、待发货、已发货、已完成等状态,这里我们就来模拟一下一个订单完成的全状态。

我们首先定义抽象状态,以及其相关实现:

java 复制代码
public interface OrderState {
    void handle();
}
java 复制代码
public class PendingPaymentState implements OrderState {
    public void handle() {
        System.out.println("订单待付款,提示用户付款...");
    }
}

public class PaidState implements OrderState {
    public void handle() {
        System.out.println("订单已付款,准备发货...");
    }
}

public class ShippedState implements OrderState {
    public void handle() {
        System.out.println("订单已发货,等待确认收货...");
    }
}

public class CompletedState implements OrderState {
    public void handle() {
        System.out.println("订单已完成,谢谢惠顾!");
    }
}

其次我们定义一个环境类:

java 复制代码
public class OrderContext {
    private OrderState state;

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

    public void request() {
        if (state == null) {
            throw new IllegalStateException("订单状态未设置!");
        }
        state.handle();
    }
}

最后我们定义一个测试类:

java 复制代码
public class Main {
    public static void main(String[] args) {
        OrderContext order = new OrderContext();

        order.setState(new PendingPaymentState());
        order.request(); // 输出:订单待付款,提示用户付款...

        order.setState(new PaidState());
        order.request(); // 输出:订单已付款,准备发货...

        order.setState(new ShippedState());
        order.request(); // 输出:订单已发货,等待确认收货...

        order.setState(new CompletedState());
        order.request(); // 输出:订单已完成,谢谢惠顾!
    }
}

其实经验丰富的程序员可以联想到一个词:那就是状态机,我们也可以对上述的案例进行进一步的加强,比如可以让每一个状态都拥有切换状态的行为,这样环境类可以直接形成一个状态流程,可以通过调用该状态的方法就可以实现与其他状态的关联。

3. 应用场景

  • 流程框架
  • 状态机
  • ......

4. 总结

使用状态模式可以让我们的业务逻辑更加的清晰,并且具有更高的扩展性,同时符合开闭原则,可以让我们业务代码避免过于臃肿。但是缺点在于会增加我们类的数量,我们业务每次增加一个状态就会添加一个新的状态实现类,但是其实比起if...else还是更加优雅的。

其实状态模式用好了对于我们开发人员来说是跟提高我们代码的可读性,比如目前我所工作的项目中就有应用到,我也深刻体会到了状态模式扩展性的作用有多高。

相关推荐
JAVA学习通1 小时前
励志从零打造LeetCode平台之C端竞赛列表
java·vscode·leetcode·docker·状态模式
Chengbei116 小时前
某211高校从一个文档到十八万条sfz泄露和命令执行
人工智能·安全·web安全·网络安全·系统安全·状态模式·安全架构
kyriewen117 小时前
代码写成一锅粥?这5种设计模式让你的项目“起死回生”
前端·javascript·设计模式·typescript·ecmascript·html5
kyriewen7 小时前
代码写成一锅粥?这5种设计模式让你的项目“起死回生”
前端·javascript·设计模式
两年半的个人练习生^_^8 小时前
每日一学:设计模式之原型模式
java·开发语言·设计模式·原型模式
前端不太难8 小时前
养门槛高、成本难控:OpenClaw的“好用”与“难用”
状态模式·openclaw
断眉的派大星1 天前
工厂模式(Factory Pattern)完整详解
python·设计模式
AI大法师1 天前
复盘 TikTok 品牌升级:动态品牌系统应该怎么理解和落地
大数据·人工智能·设计模式
前端不太难1 天前
当 AI 出错时,责任在谁?系统设计中的责任归属(Accountability)
人工智能·状态模式
妙蛙种子3111 天前
【Java设计模式 | 创建者模式】建造者模式
java·开发语言·后端·设计模式·建造者模式