介绍Java 状态设计模式

介绍Java 状态设计模式

本文我们介绍GoF中一个行为设计模式————状态模式。首先总体介绍其目的,解释其能解决什么问题。然后看下状态模式的UML图并通过实际示例进行实现。

概要说明

状态设计模式的主要目的是在不改变类的情况下改变对象的行为。同时,实现该模式代码需保持简洁,不能有很多if/else语句。

加入我们有一个包裹需要邮寄。包裹本身可以被订购,然后送到邮局,最后由客户接收。现在需要根据实际状态打印投递过程的状态。

最简单的方法增加一些布尔标志,然后在类的每个方法中使用简单if/else语句。在这个简单场景中不会很复杂,然而当有更多状态需要处理时可能会变得复杂并污染代码。

另外,每个状态的业务逻辑将分散在不同方法中。这时可以考虑使用状态模式进行优化。通过使用状态模式,我们在决策类中封装逻辑,符合单一责任原则和开闭原则,让代码更简洁、更易维护。

UML 图

uml

在uml图中,我们看到Context类与程序执行过程中将被改变的State进行关联。

上下文委托行为给状态实现。也就是说,所有传入的请求都将由状态的具体实现来处理。我们看到状态和逻辑是分离的,添加新状态很简单——仅根据需要添加另一个状态实现。

代码实现

下面开始设计应用。前面提及包裹有预定、投递和已收状态。因此,我们有三个状态和上下文类。

首先,我们定义context类,也就是我们的Package类:

public class Package {
 
    private PackageState state = new OrderedState();
 
    // getter, setter
 
    public void previousState() {
        state.prev(this);
    }
 
    public void nextState() {
        state.next(this);
    }
 
    public void printStatus() {
        state.printStatus();
    }
}

我们看到包含一个状态的引用,同时 previousState(), nextState() 和 printStatus() 三个方法委托工作给状态对象。状态被彼此链接,每个状态在两个方法中将基于传入状态对象被设置为另一种状态。

客户端与Package类进行交互,无需处理并设置状态,所有的客户端仅需要做的是调用两个方法进行前进或后退。接下来我们实现PackageState接口和其三个实现:

public interface PackageState {
 
    void next(Package pkg);
    void prev(Package pkg);
    void printStatus();
}

该接口需被每个具体状态类进行实现。首先我们实现OrderState:

public class OrderedState implements PackageState {
 
    @Override
    public void next(Package pkg) {
        pkg.setState(new DeliveredState());
    }
 
    @Override
    public void prev(Package pkg) {
        System.out.println("The package is in its root state.");
    }
 
    @Override
    public void printStatus() {
        System.out.println("Package ordered, not delivered to the office yet.");
    }
}

当包裹被预定后,next方法改变其状态为下一个状态。预定状态被显示标记为初始状态,我们看到两个方法如何改变状态。

下面看DeliveredState 类:

public class DeliveredState implements PackageState {
 
    @Override
    public void next(Package pkg) {
        pkg.setState(new ReceivedState());
    }
 
    @Override
    public void prev(Package pkg) {
        pkg.setState(new OrderedState());
    }
 
    @Override
    public void printStatus() {
        System.out.println("Package delivered to post office, not received yet.");
    }
}

我们有看到了状态之间的链接。包裹状态从预定状态在不断变化,printStatus方法保持同样变化。
最后是ReceivedState类:

public class ReceivedState implements PackageState {
 
    @Override
    public void next(Package pkg) {
        System.out.println("This package is already received by a client.");
    }
 
    @Override
    public void prev(Package pkg) {
        pkg.setState(new DeliveredState());
    }
}

因为已经是最后状态,仅能回滚至前一个状态。
我们已经看到了这样设计的好处,因为一个状态知道另一个状态,它们彼此紧密关联。

测试

下面通过测试看看如何实现流转。首先让我们验证一下状态转换是否如预期一样:

@Test
public void givenNewPackage_whenPackageReceived_thenStateReceived() {
    Package pkg = new Package();
 
    assertThat(pkg.getState(), instanceOf(OrderedState.class));
    pkg.nextState();
 
    assertThat(pkg.getState(), instanceOf(DeliveredState.class));
    pkg.nextState();
 
    assertThat(pkg.getState(), instanceOf(ReceivedState.class));
}

然后,我们检查下包裹状态是否可以回滚:

@Test
public void givenDeliveredPackage_whenPrevState_thenStateOrdered() {
    Package pkg = new Package();
    pkg.setState(new DeliveredState());
    pkg.previousState();
 
    assertThat(pkg.getState(), instanceOf(OrderedState.class));
}

下面验证状态改变时, printStatus() 方法输出结果是否正确:

public class StateDemo {
 
    public static void main(String[] args) {
 
        Package pkg = new Package();
        pkg.printStatus();
 
        pkg.nextState();
        pkg.printStatus();
 
        pkg.nextState();
        pkg.printStatus();
 
        pkg.nextState();
        pkg.printStatus();
    }
}

输出结果如下:

Package ordered, not delivered to the office yet.
Package delivered to post office, not received yet.
Package was received by client.
This package is already received by a client.
Package was received by client.

当我们一直在改变上下文的状态时,行为在改变,但是类保持不变,使用的API也一样。
而且,状态之间的转换已经发生,我们类改变了状态并因此改变了其行为。

缺点

状态模式的缺点是需要实现状态转换。这导致状态硬编码,一般情况不是好的习惯。但这取决于你的需求,可能也不是问题。

状态 VS 策略

两个模式比较类似,但UML图相同,其背后思想却有差异。
首先,策略模式定义了一组可互换的算法。通常它们实现相同目的,但有不同实现。如排序或渲染算法。状态模式中行为可能根据实际状态而完全改变。
其次,策略模式中客户端必须知道可以显式使用和更改的策略。而状态模式中每个状态链接到另一个状态,就像在有限状态机中自动流转。

总结

状态设计模式在避免过多使用if/else语句是非常有用。其把业务逻辑抽取至独立类中,上下文对象代理状态类的方法行为。此外,我们可以利用状态之间的转换从而更改上下文的状态。

通常该设计模式在简单应用中使用较好,更好的方法可以看下Spring State Machine。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值