深入理解Java设计模式——状态模式

一、什么是状态模式

状态模式允许一个对象在其内部状态改变的时候改变其行为。这个对象看上去就像是改变了它的类一样。

二、状态模式应用场景

  1. 一个对象的行为取决于它的状态,并且它必须在运行时刻根据状态改变它的行为。
  2. 操作中含有庞大的多分支的条件语句,且这些分支依赖于该对象的状态。这个状态通常用一个或多个枚举常量表示。 通常,有多个操作包含这一相同的条件结构。State模式将每一个条件分支放入一个独立的类中。这使得你可以根据对象自身的情况将对象的状态作为一个对象,这一对象可以不依赖于其他对象而独立变化。

三、状态模式实现

1. 需要重构的代码

public String orderState(String state) {
        if (state.equals("0")) {
            return "已经发货";
        }
        if (state.equals("1")) {
            return "正在运送中...调用第三方快递接口 展示 运送信息";
        }
        if (state.equals("2")) {
            return "正在派送中... 返回派送人员信息";
        }
        if (state.equals("3")) {
            return "已经签收,提示给用户快递员评价";
        }
        if (state.equals("4")) {
            return "拒绝签收, 重新开始申请退单";
        }
        if (state.equals("5")) {
            return "订单交易失败,调用短信接口提示 ";
        }
        return "未找到对应的状态";
    }
}


四、状态模式与策略模式区别

在这里插入图片描述

1、状态模式重点在各状态之间的切换从而做不同的事情,而策略模式更侧重于根据具体情况选择策略,并不涉及切换。
2、状态模式不同状态下做的事情不同,而策略模式做的都是同一件事,例如聚合支付平台,有支付宝、微信支付、银联支付,虽然策略不同,但最终做的事情都是支付,也就是说他们之间是可替换的。反观状态模式,各个状态的同一方法做的是不同的事,不能互相替换。
状态模式封装了对象的状态,而策略模式封装算法或策略。因为状态是跟对象密切相关的,它不能被重用;而通过从Context中分离出策略或算法,我们可以重用它们。
在状态模式中,每个状态通过持有Context的引用,来实现状态转移;但是每个策略都不持有Context的引用,它们只是被Context使用。

排序算法 快速排序 选择排序 冒泡排序 策略
只要是和状态相关 订单、交易、支付状态。

五、状态模式实现

maven:

    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.0.1.RELEASE</version>
    </parent>
    <dependencies>
        <!-- sprinboot web -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <version>1.16.10</version>
        </dependency>
        <dependency>
            <groupId>commons-lang</groupId>
            <artifactId>commons-lang</artifactId>
            <version>2.6</version>
        </dependency>

    </dependencies>

启动类:

package com.demo.order;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;

@SpringBootApplication
public class AppOrderService {
    public static void main(String[] args) {
        SpringApplication.run(AppOrderService.class);
    }
}

utils:

package com.demo.order.utils;

import org.springframework.beans.BeansException;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.stereotype.Component;

@Component
public class SpringUtils implements ApplicationContextAware {

    private static ApplicationContext applicationContext;

    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        this.applicationContext = applicationContext;
    }

    // 获取applicationContext
    public static ApplicationContext getApplicationContext() {
        return applicationContext;
    }

    // 通过name获取Bean
    public static Object getBean(String name) {
        return getApplicationContext().getBean(name);
    }

    // 通过class获取Bean
    public static <T> T getBean(Class<T> clazz) {
        return getApplicationContext().getBean(clazz);
    }

    // 通过name,以及Clazz返回指定的Bean
    public static <T> T getBean(String name, Class<T> clazz) {
        return getApplicationContext().getBean(name, clazz);
    }
}

package com.demo.order.service;

public interface OrderState {
    /**
     * 返return
     */
    Object orderService();
}

package com.demo.order.service.impl;

import com.demo.order.service.OrderState;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Component;

@Slf4j
@Component
public class AlreadySignedOrderState implements OrderState {
    @Override
    public Object orderService() {
        log.info(">> 切换已经签收状态..");
        return "切换已经签收状态";
    }
}

package com.demo.order.service.impl;

import com.demo.order.service.OrderState;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Component;

@Slf4j
@Component
public class InTransitOrderState implements OrderState {
    @Override
    public Object orderService() {
        log.info(">> 切换为正在运送状态..");
        return "切换为正在运送状态";
    }
}

package com.demo.order.service.impl;

import com.demo.order.service.OrderState;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Component;

@Slf4j
@Component
public class ShippedAlreadyOrderState implements OrderState {
    @Override
    public Object orderService() {
        log.info(">> 切换为已经发货状态..");
        return "切换为已经发货状态";
    }
}

package com.demo.order.context;

import com.demo.order.service.OrderState;

public class StateContext {
    private OrderState orderState;

    public StateContext(OrderState orderState) {
        this.orderState = orderState;
    }

    public void switchStateOrder() {
        orderState.orderService();
    }
}

package com.demo.order;

import com.demo.order.context.StateContext;
import com.demo.order.service.impl.ShippedAlreadyOrderState;

public class Test {
    public static void main(String[] args) {
        StateContext stateContext = new StateContext(new ShippedAlreadyOrderState());
        stateContext.switchStateOrder();
    }
}

package com.demo.order.controller;

import com.demo.order.context.StateContext;
import com.demo.order.service.OrderState;
import com.demo.order.utils.SpringUtils;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

@RestController
public class OrderController {

    @RequestMapping("/order")
    public String order(String stateBeanId) {
        // 1.使用Spring上下文获取bean对象
        OrderState orderState = SpringUtils.getBean(stateBeanId, OrderState.class);
        StateContext stateContext = new StateContext(orderState);
        stateContext.switchStateOrder();
        return "success";
    }
}

1.运行项目
在这里插入图片描述
2.分别访问:
http://127.0.0.1:8080/order?stateBeanId=alreadySignedOrderState
http://127.0.0.1:8080/order?stateBeanId=inTransitOrderState
http://127.0.0.1:8080/order?stateBeanId=shippedAlreadyOrderState
在这里插入图片描述
在这里插入图片描述

  • 6
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 3
    评论
当谈到 Java 设计模式时,有一些常见的设计模式是值得深入了解的。设计模式是一套经过验证的解决问题的经验法则,可以帮助我们在软件开发中构建可维护、可扩展和可复用的代码。 以下是一些常见的 Java 设计模式: 1. 创建型设计模式: - 单例模式(Singleton):确保一个类只有一个实例,并提供全局访问点。 - 工厂模式(Factory):定义一个用于创建对象的接口,但具体类的决定由子类决定。 - 建造者模式(Builder):将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示。 2. 结构型设计模式: - 适配器模式(Adapter):将一个类的接口转换成客户希望的另一个接口。 - 装饰器模式(Decorator):动态地将责任附加到对象上,若要扩展功能,装饰器提供了比继承更加有弹性的替代方案。 - 组合模式(Composite):将对象组合成树形结构以表示“部分-整体”的层次结构。 3. 行为型设计模式: - 观察者模式(Observer):定义了对象之间的一对多依赖关系,当一个对象状态发生改变时,其依赖的对象都会收到通知。 - 策略模式(Strategy):定义了算法族,分别封装起来,使得他们之间可以相互替换,使得算法的变化独立于使用算法的客户。 - 命令模式(Command):将请求封装成对象,使得可以用不同的请求对客户进行参数化。 这些只是一小部分常见的设计模式,每个模式都有特定的用途和优势。深入了解这些模式,并将其应用到你的项目中,可以提高代码的可读性、可维护性和可扩展性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

超级码里喵

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值