背景

在公司经常会遇到一些需要做一连串相似的业务逻辑判断需求,比如使用不同支付方式实现商品打折、商店根据客户的vip等级给予客户不同的优惠政策之类的业务场景。假设现在我们接到一个需求,根据客户的VIP等级,给与不同的优惠政策:
// VIP1 消费1000 ~ 2000,9折优惠, 消费2000~5000,8折优惠,消费5000~10000,7折优惠, 消费10000~,6折优惠
    static String VIP1 = "VIP1";

    // VIP2 消费1000 ~ 2000,8.5折优惠, 消费2000~5000,7.5折优惠,消费5000~10000,6.5折优惠,消费10000~,5.5折优惠
    static String VIP2 = "VIP2";

    // VIP3 消费1000 ~ 2000,8折优惠, 消费2000~5000,7折优惠,消费5000~10000,6折优惠, 消费10000~,5折优惠
    static String VIP3 = "VIP3";

    // VIP4 消费1000 ~ 2000,7.5折优惠, 消费2000~5000,6.5折优惠,消费5000~10000,5.5折优惠, 消费10000~,4.5折优惠
    static String VIP4 = "VIP4";

    // VIP5 消费1000 ~ 2000,7折优惠, 消费2000~5000,6折优惠,消费5000~10000,5折优惠, 消费10000~,4折优惠
    static String VIP5 = "VIP5";

    // VIP6 消费1000 ~ 2000,6.5折优惠, 消费2000~5000,5.5折优惠,消费5000~10000,4.5折优惠, 消费10000~,3.5折优惠
    static String VIP6 = "VIP6";

if-else

现在商店来了一个客户,我们需要根据客户的会员等级实现不同的优惠策略。通常使用if-else代码如下:

	static BigDecimal PRICE1000 = new BigDecimal(1000);
    static BigDecimal PRICE2000 = new BigDecimal(2000);
    static BigDecimal PRICE5000 = new BigDecimal(5000);
    static BigDecimal PRICE10000 = new BigDecimal(10000);
	public BigDecimal getPrice(Client client, BigDecimal price){
        BigDecimal realPrice = price;
        if(VIP1.equals(client.getType())){
            if(PRICE1000.compareTo(price) < 0 && PRICE2000.compareTo(price) >= 0){
                realPrice = new BigDecimal("0.9").multiply(price);
            } else if(PRICE2000.compareTo(price) < 0 && PRICE5000.compareTo(price) >= 0){
                realPrice = new BigDecimal("0.8").multiply(price);
            } else if(PRICE5000.compareTo(price) < 0 && PRICE10000.compareTo(price) >= 0){
                realPrice = new BigDecimal("0.7").multiply(price);
            } else if(PRICE10000.compareTo(price) < 0){
                realPrice = new BigDecimal("0.6").multiply(price);
            }
        } else if(VIP2.equals(client.getType())){
            //...
        } else if(VIP3.equals(client.getType())){
            //...
        } else if(VIP4.equals(client.getType())){
            //...
        } else if(VIP5.equals(client.getType())){
            //...
        } else if(VIP6.equals(client.getType())){
            //...
        }
        return realPrice;
    }

策略模式

很抱歉,实在写不下去了,重复这样子写有些累,后面VIP2~VIP6只需要仿照VIP1的优惠策略写就行,我在这里只是想要表达这种写法实在不怎么样,读者明白我的意思就行。这里可以通过使用策略模式进行优化:

// 策略接口
public interface VIP {
    static BigDecimal PRICE1000 = new BigDecimal(1000);
    static BigDecimal PRICE2000 = new BigDecimal(2000);
    static BigDecimal PRICE5000 = new BigDecimal(5000);
    static BigDecimal PRICE10000 = new BigDecimal(10000);
    BigDecimal getPrice(BigDecimal price);
}

// 具体策略实现类
public class VIP1 implements VIP {
    @Override
    public BigDecimal getPrice(BigDecimal price) {
        BigDecimal realPrice = price;
        if(PRICE1000.compareTo(price) < 0 && PRICE2000.compareTo(price) >= 0){
            realPrice = new BigDecimal("0.9").multiply(price);
        } else if(PRICE2000.compareTo(price) < 0 && PRICE5000.compareTo(price) >= 0){
            realPrice = new BigDecimal("0.8").multiply(price);
        } else if(PRICE5000.compareTo(price) < 0 && PRICE10000.compareTo(price) >= 0){
            realPrice = new BigDecimal("0.7").multiply(price);
        } else if(PRICE10000.compareTo(price) < 0){
            realPrice = new BigDecimal("0.6").multiply(price);
        }
        return realPrice;
    }
}
// 这里只将VIP1的具体实现放了上来,其他几种策略意思差不多

public class Shop {
	public BigDecimal getPrice(Client client, BigDecimal price){
        VIP vip = null;
        if(VIP1.equals(client.getType())){
            vip = new VIP1();
        } else if(VIP2.equals(client.getType())){
            vip = new VIP2();
        } else if(VIP3.equals(client.getType())){
            vip = new VIP3();
        } else if(VIP4.equals(client.getType())){
            vip = new VIP4();
        } else if(VIP5.equals(client.getType())){
            vip = new VIP5();
        } else if(VIP6.equals(client.getType())){
            vip = new VIP6();
        }
        return vip.getPrice(price);
    }
	public static void main(String[] args) {
        Shop shop = new Shop();
        Client client = new Client();
        client.setType(Shop.VIP1);
        client.shopping(shop, new BigDecimal(10000));
    }
}

策略模式+工厂模式

显然,这样子代码看着舒服了一些,而且后面如果哪个VIP等级的优惠政策有了更改,我们只需要在相应的策略实现类中进行修改即可,另外之后如果添加VIP7…VIPN是只需要添加具体策略实现类就行。当然,这样也会导致后面的类文件十分多,会增添项目文件管理上的麻烦,这一点是需要我们注意的。另外,眼尖的你应该也发现了,上面的代码还是有很多if-else,具体策略类中的if-else暂时无能为力(可以使用策略模式,但是感觉使用了策略模式之后项目文件管理会变成一个巨大的麻烦),后面如果我有了方法解决这个问题,会再更新这篇文章。具体使用哪种策略时使用的if-else判断是可以利用Map来替换,Map中的key为用户VIP类型,value为具体策略类型。利用Map我们可以直接通过get方法拿到我们想要的具体策略实现类的实例,从而获取相应的方法。具体实现大体如下:

public class VIPFactory {
    private static VIPFactory factory = new VIPFactory();
    private VIPFactory(){}
    private static Map vipMap = new HashMap<>();
    static {
        vipMap.put(Shop.VIP1, new VIP1());
        vipMap.put(Shop.VIP2, new VIP2());
        vipMap.put(Shop.VIP3, new VIP3());
        vipMap.put(Shop.VIP4, new VIP4());
        vipMap.put(Shop.VIP5, new VIP5());
        vipMap.put(Shop.VIP6, new VIP6());
    }
    public static VIPFactory getInstance(){
        return factory;
    }
    public VIP getVip(String vipType){
        return (VIP) vipMap.get(vipType);
    }
}

public class Shop {
    public BigDecimal getPrice(Client client, BigDecimal price){
        VIP vip = VIPFactory.getInstance().getVip(client.getType());
        return vip.getPrice(price);
    }
    public static void main(String[] args) {
        Shop shop = new Shop();
        Client client = new Client();
        client.setType(Shop.VIP1);
        client.shopping(shop, new BigDecimal(10000));
    }
}

具体的策略实现类我就不再赘述了,工厂使用的是单例模式。其实这个时候可以很明显到感觉代码看起来舒服多了。