什么是设计模式?
今天我们来说策略模式,何为策略模式?让我们先看看以下示例代码
这里我们拿支付方式来举例
工程结构如下:
这是我们的支付接口,定义了一个支付的方法 根据不同的支付类型给予不同的优惠
public interface Pay {
/**
* @param skuPrice 提交的金额
* @return
*/
BigDecimal payment(Integer,BigDecimal skuPrice);
}
我们来看看这个支付接口的实现类
public class PayService implements Pay {
public BigDecimal payment(Integer type,BigDecimal skuPrice) {
// 1. 支付宝 优惠0.7元
if (1 == type) {
return skuPrice.subtract(new BigDecimal(0.7));
}
// 2. 微信支付 优惠0.3
if (2 == type) {
return skuPrice.subtract(new BigDecimal(0.3));
}
return skuPrice;//否则银行卡支付原价
}
}
这种方式看似没有什么问题,想要的功能也能为你实现,但一旦优惠条件的算法难度上升导致代码量剧增,大量的if else语句会使整个程序代码看起来又长又臭,而且对于产品的扩展性极其不友好
思考:如何避免使用if语句并能使程序根据不同的支付类型给予不同的优惠?
为什么需要策略模式?
切入正题!改造我们的代码
首先,使用不同的支付方式实现支付接口
public class WeChatPay implements Pay {
/**
* 微信支付10元+ 优惠0.7
* @param skuPrice 提交的金额
* @return
*/
public BigDecimal payment(BigDecimal skuPrice) {
//大于10元就优惠
if (skuPrice.compareTo(new BigDecimal(10)) >= 0)
return skuPrice.subtract(new BigDecimal(0.7));//返回优惠价
return skuPrice;//返回原价
}
}
public class AliPay implements Pay {
public BigDecimal payment(BigDecimal skuPrice) {
if (skuPrice.compareTo(new BigDecimal(10)) >= 0)
return skuPrice.subtract(new BigDecimal(0.3));//返回优惠价
return skuPrice;
}
}
public class BankPay implements Pay {
/**
* @param skuPrice 提交的金额
* @return
*/
public BigDecimal payment(BigDecimal skuPrice) {
return skuPrice;//银行卡支付无优惠
}
}
策略类
public class PayStrategy<T> {
private Pay pay;
public PayStrategy (Integer i) {
if (i == 1) {
pay= new WeChatPay();
} else if (i == 2) {
pay = new AliPay();
} else {
pay = new BankPay();
}
}
public BigDecimal payment(BigDecimal skuPrice) {
return pay.payment(skuPrice);
}
}
来康康测试结果
@Test
public void test() {
PayStrategy pay = new PayStrategy(1);//1=微信支付
BigDecimal bigDecimal = pay.payment(new BigDecimal(20));
logger.info("测试结果:支付金额 {}",bigDecimal);
}
经测试我们的策略模式成功了,但是大家有没有发现个问题在我们的策略类中有一段这样的代码
这是啥玩意? 如果按照这种写法 如果产品多增加一种支付类型那么就意味着你要再增加if else! 然后不停的new哦~ 嘿嘿嘿. 这非常的龌龊!
作为俺们村的村草我是绝对不允许这种事情发生 咳咳~
我们可以把创建实例的工作交个Spring 大大简化开发,让我们看下面的代码
新建一个支付类型枚举
public enum PayEnum {
WeChatPay,
PayStrategy,
AliPay;
}
在我们的支付接口实现类上标注@component注解 将对象加载进入Springbean中 起别名为类名 例如:
@Component("WeChatPay")
public class WeChatPay implements Pay {
@Component("BankPay")
public class BankPay implements Pay {
@Component("AliPay")
public class AliPay implements Pay {
改造策略类
@Component
public class PayStrategy {
private Pay pay;
private Map<String,Pay> map;
public void setStrategy(PayEnum strategy) {
pay = map.get(strategy.toString());
System.out.println("pay == " + pay);
}
/**
* 工厂初始化注入map
*/
public PayStrategy(Map<String,Pay> map) {
this.map = map;
}
public BigDecimal payment(BigDecimal sku) {
return pay.payment(sku);
}
}
策略工厂,说明一下该方法的作用,可见下列方法的有一个map参数,它是如何注入的呢?
@Component
public class PayStrategyFactory {
@Bean
public PayStrategy payStrategy(Map<String,Pay> map) {
return new PayStrategy(map);
}
}
因为spring的默认机制,默认是单例模式 我们并不需要担心map是否会失效,在策略类中定义的map为我们实现了实现类的单例 让我们来测试一下
@ComponentScan
public class Application {
public static void main(String[] args) {
ApplicationContext ctx = new AnnotationConfigApplicationContext(Application.class);//初始化IOC容器
try {
PayStrategy payStrategy = ctx.getBean(PayStrategy.class);
payStrategy.setStrategy(PayEnum.WeChatPay);
System.out.println("支付价格为 == " + payStrategy.payment(new BigDecimal(20)).setScale(2,BigDecimal.ROUND_DOWN));
payStrategy.setStrategy(PayEnum.AliPay);
System.out.println("支付价格为 == " + payStrategy.payment(new BigDecimal(20)).setScale(2,BigDecimal.ROUND_DOWN));
payStrategy.setStrategy(PayEnum.WeChatPay);
System.out.println("支付价格为 == " + payStrategy.payment(new BigDecimal(20)).setScale(2,BigDecimal.ROUND_DOWN));
payStrategy.setStrategy(PayEnum.BankPay);
System.out.println("支付价格为 == " + payStrategy.payment(new BigDecimal(20)).setScale(2,BigDecimal.ROUND_DOWN));
} catch (Exception e) {
e.printStackTrace();
}
}
}
可以看到我们在使用相同枚举时候 拿到的都是同一个对象,这就完成了单例模式
同学,你学废了嘛,设计模式深撸起来会感觉非常奇妙,在上述代码中还有些许不完美,让我们一起进步吧,这片文章要是对你有帮助记得点个赞哟
每日进步一点点,Day Day Up