优化if-else的几种方式

  • 策略模式
  • 1、创建支付策略接口
  • 2、书写不同的支付方式逻辑代码
  • 微信支付
  • QQ支付
  • 3、service层的实现类使用
  • 4、controller层的调用
  • 说明
  • 枚举与策略模式结合
  • 1、创建枚举
  • 2、service层书写处理方法
  • 3、controller层调用
  • 4、说明
  • Lambda表达式与函数接口
  • 说明


策略模式

策略模式允许在运行时选择算法。策略模式是将算法定义成独立的类,并在运行时动态选择要使用的具体的算法,以此来避免多个if-else或switch语句的使用。
下面以支付功能为例子进行说明。
假设我们有一个支付系统,支持微信、QQ等多种支付方式。用户在支付时会选择自己需要的支付方式,后台功能接口中接收到用户的支付方式选择时,会进行不同的处理。在这里会产生if-else或者switch。如何使用策略模式来消除这些if-else的使用,下面示例说明。

1、创建支付策略接口

/**
 * 支付策略
 */
public interface PaymentStrategy {

    void pay(double amount);

}

2、书写不同的支付方式逻辑代码

微信支付

import org.springframework.stereotype.Component;

@Component
public class WeiXinPayment implements PaymentStrategy {

    @Override
    public void pay(double amount) {
        System.out.println("微信支付" + amount);
    }
}

QQ支付

import org.springframework.stereotype.Component;

@Component
public class QQPayment implements PaymentStrategy {

    @Override
    public void pay(double amount) {
        System.out.println("QQ支付" + amount);
    }
}

3、service层的实现类使用

import com.hysoft.study.service.PaymentStrategy;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import java.util.List;
import java.util.Map;
import java.util.function.Function;
import java.util.stream.Collectors;

/**
 * 支付实现类
 */
@Service
public class PaymentServiceImpl {

    private final Map<String, PaymentStrategy> strategies;

    @Autowired
    public PaymentServiceImpl(List<PaymentStrategy> paymentStrategies){
        this.strategies = paymentStrategies.stream()
                .collect(Collectors.toMap(s -> s.getClass().getSimpleName().toLowerCase(), Function.identity()));
    }

    public void processPayment(String strategyName,double amount){
        PaymentStrategy strategy = strategies.getOrDefault(strategyName,null);
        if (strategy != null){
            strategy.pay(amount);
        }else {
            throw new IllegalArgumentException("Strategy not found" + strategyName);
        }
    }

}

4、controller层的调用

import com.hysoft.study.service.impl.PaymentServiceImpl;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

@RestController
@RequestMapping("payment")
public class PaymentController {

    @Autowired
    private PaymentServiceImpl paymentService;

    @PostMapping("test")
    public void test(String paymentname,double amount){
        this.paymentService.processPayment(paymentname,amount);
    }

}

说明

需要注意的是,controller层调用service层实现方法的适合,传递了参数paymentname(支付方式),这个支付方式需要提前和前端调用人员协商好,这里的名称是各种支付方式的bean名称,在service处理中已经有所体现

优化if-else的几种方式_策略模式


因此在这里名称传值可以是qqpayment或weixinpayment。因此传值需要提前和前端进行协商。

枚举与策略模式结合

枚举类型不仅可以用来表示一组常量,还可以定义与这些常量相关联的行为。结合策略模式,可以进一步简化代码。

1、创建枚举

public enum OrderStatus {

    NEW {
        @Override
        public void process () {
            System.out.println("处理新建订单");
        }
    },

    PAID {
        @Override
        public void process () {
            System.out.println("订单已支付");
        }
    },

    UNPAD {
        @Override
        public void process() {
            System.out.println("订单未支付");
        }
    };

    public abstract void process();
}

2、service层书写处理方法

import com.hysoft.study.model.OrderStatus;
import org.springframework.stereotype.Service;

@Service
public class OrderServiceImpl {

    public void handleOrder(OrderStatus status) {
        status.process();
    }

}

3、controller层调用

import com.hysoft.study.model.OrderStatus;
import com.hysoft.study.service.impl.OrderServiceImpl;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

@RestController
@RequestMapping("order")
public class OrderController {

    @Autowired
    private OrderServiceImpl orderService;

    @PostMapping("test")
    public void test(String status){
        OrderStatus aNew = OrderStatus.valueOf(status);
        this.orderService.handleOrder(aNew);
    }

}

4、说明

controller层调用时传输的参数status即使在枚举中的各常量,例如NEW或者PAID、UNPAD等

Lambda表达式与函数接口

以下示例在service层实现类中直接书写了不同vip等级的结算金额的逻辑
import org.springframework.stereotype.Service;
import java.util.HashMap;
import java.util.Map;
import java.util.function.Function;

@Service
public class StreamServiceImpl {

    private final Map<String, Function<Double, Double>> discountFunctions = new HashMap<>();

    public StreamServiceImpl() {
        discountFunctions.put("VIP1", e -> e * 0.95);
        discountFunctions.put("VIP2", e -> e * 0.95 - 20);
    }

    public double applyDiscount(String vipname, double price) {
        Double apply = discountFunctions.getOrDefault(vipname, Function.identity()).apply(price);
        return apply;
    }
}

在controller层调用时,需要传入vip等级和总计算金额,计算结果时打折后金额

import com.hysoft.study.service.impl.StreamServiceImpl;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

@RestController
@RequestMapping("stream")
public class StreamController {

    @Autowired
    private StreamServiceImpl streamService;

    @PostMapping("test")
    public double test(String vipname,Double price){
        return this.streamService.applyDiscount(vipname,price);
    }

}

说明

传参vipname的值应参考service层类中的vip参数

优化if-else的几种方式_策略模式_02