前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >专栏 >策略模式(Strategy Pattern)深度解析教程

策略模式(Strategy Pattern)深度解析教程

原创
作者头像
用户11531739
发布2025-03-17 17:20:07
发布2025-03-17 17:20:07
920
举报
一、模式定义

策略模式属于行为型设计模式,通过定义算法族并将其封装为独立的策略类,使得算法可以动态切换且与使用它的客户端解耦。该模式通过组合替代继承,符合开闭原则(对扩展开放,对修改关闭)。

二、核心角色
  1. Strategy(策略接口)
    • 定义所有支持的算法的公共接口
  2. ConcreteStrategy(具体策略)
    • 实现策略接口的具体算法
  3. Context(上下文)
    • 持有策略引用,提供修改策略的方法
    • 将客户端请求委托给当前策略
三、经典实现(电商促销场景)
代码语言:java
复制
// 1. 策略接口
public interface DiscountStrategy {
    double applyDiscount(double originalPrice);
}

// 2. 具体策略实现
// 无折扣策略
public class NoDiscountStrategy implements DiscountStrategy {
    @Override
    public double applyDiscount(double originalPrice) {
        return originalPrice;
    }
}

// 满减策略
public class FullReductionStrategy implements DiscountStrategy {
    private final double fullAmount;
    private final double reduction;

    public FullReductionStrategy(double fullAmount, double reduction) {
        this.fullAmount = fullAmount;
        this.reduction = reduction;
    }

    @Override
    public double applyDiscount(double originalPrice) {
        return originalPrice >= fullAmount ? 
            originalPrice - reduction : originalPrice;
    }
}

// 百分比折扣策略
public class PercentageDiscountStrategy implements DiscountStrategy {
    private final double percentage;

    public PercentageDiscountStrategy(double percentage) {
        this.percentage = percentage;
    }

    @Override
    public double applyDiscount(double originalPrice) {
        return originalPrice * (1 - percentage);
    }
}

// 3. 上下文类
public class PricingContext {
    private DiscountStrategy strategy;

    public PricingContext(DiscountStrategy strategy) {
        this.strategy = strategy;
    }

    public void setStrategy(DiscountStrategy strategy) {
        this.strategy = strategy;
    }

    public double calculatePrice(double originalPrice) {
        return strategy.applyDiscount(originalPrice);
    }
}

// 4. 客户端使用
public class ECommerceApp {
    public static void main(String[] args) {
        // 初始策略:无折扣
        PricingContext context = new PricingContext(new NoDiscountStrategy());
        double price = 500.0;

        System.out.println("原价: " + price);
        System.out.println("默认策略价格: " + context.calculatePrice(price));

        // 切换为满减策略(满400减100)
        context.setStrategy(new FullReductionStrategy(400, 100));
        System.out.println("满减策略价格: " + context.calculatePrice(price));

        // 切换为百分比折扣(8折)
        context.setStrategy(new PercentageDiscountStrategy(0.2));
        System.out.println("百分比折扣价格: " + context.calculatePrice(price));
    }
}

/* 输出:
原价: 500.0
默认策略价格: 500.0
满减策略价格: 400.0
百分比折扣价格: 400.0
*/
四、模式结构UML
代码语言:java
复制
          _________________________
         |       Strategy         |
         |------------------------|
         | + executeAlgorithm()   |
         |________________________|
                    ▲
       ____________|_____________
      |            |            |
______▼______  ____▼______  _____▼______
| Concrete  | | Concrete  | | Concrete  |
| StrategyA | | StrategyB | | StrategyC |
|___________| |___________| |___________|
                    ▲
                    |
               _____▼_____
              |  Context  |
              |-----------|
              | - strategy|
              |___________|
五、模式优劣分析

优势:

  • 避免使用多重条件判断语句
  • 符合开闭原则,新增策略无需修改现有代码
  • 算法可自由切换和组合
  • 便于单元测试(每个策略可独立测试)

劣势:

  • 客户端必须了解不同策略的区别
  • 策略类数量可能膨胀(需配合工厂模式管理)
  • 增加对象数量(每个策略都是独立对象)
六、应用场景
  1. 支付方式选择(支付宝、微信、银行卡等)
  2. 排序算法切换(快速排序、归并排序、冒泡排序)
  3. 文件解析策略(JSON、XML、CSV解析器)
  4. 导航策略(步行导航、驾车导航、公共交通)
  5. 压缩算法选择(ZIP、RAR、7z压缩策略)
七、Java标准库应用
  • Comparator接口
代码语言:java
复制
// 策略模式在排序中的应用
List<Integer> numbers = Arrays.asList(3, 1, 4, 1, 5);

// 使用不同排序策略
Collections.sort(numbers, Comparator.naturalOrder());  // 升序策略
Collections.sort(numbers, Comparator.reverseOrder()); // 降序策略
  • ThreadPoolExecutor拒绝策略
代码语言:java
复制
// 线程池拒绝策略实现
ThreadPoolExecutor executor = new ThreadPoolExecutor(
    2, 4, 60, TimeUnit.SECONDS, 
    new LinkedBlockingQueue<>(10),
    new ThreadPoolExecutor.AbortPolicy()  // 拒绝策略
);
八、与相似模式对比

模式

核心区别

工厂模式

关注对象创建,策略模式关注行为选择

命令模式

封装操作请求,策略模式封装算法实现

状态模式

状态转移自动发生,策略需要显式选择

九、高级应用技巧
  • 策略工厂管理
代码语言:java
复制
public class DiscountStrategyFactory {
    private static final Map<String, DiscountStrategy> strategies = new HashMap<>();

    static {
        strategies.put("NONE", new NoDiscountStrategy());
        strategies.put("FULL_100", new FullReductionStrategy(400, 100));
        strategies.put("20%OFF", new PercentageDiscountStrategy(0.2));
    }

    public static DiscountStrategy getStrategy(String strategyKey) {
        return strategies.getOrDefault(strategyKey, new NoDiscountStrategy());
    }
}

// 使用示例
DiscountStrategy strategy = DiscountStrategyFactory.getStrategy("20%OFF");
  • 组合策略
代码语言:java
复制
// 策略组合器
public class CombinedDiscountStrategy implements DiscountStrategy {
    private final List<DiscountStrategy> strategies;

    public CombinedDiscountStrategy(DiscountStrategy... strategies) {
        this.strategies = Arrays.asList(strategies);
    }

    @Override
    public double applyDiscount(double originalPrice) {
        double currentPrice = originalPrice;
        for (DiscountStrategy strategy : strategies) {
            currentPrice = strategy.applyDiscount(currentPrice);
        }
        return currentPrice;
    }
}

// 使用示例
DiscountStrategy combined = new CombinedDiscountStrategy(
    new FullReductionStrategy(300, 50),
    new PercentageDiscountStrategy(0.1)
);
  • Spring框架集成
代码语言:java
复制
// 通过@Qualifier注入不同策略
@Service
public class PaymentService {
    private final PaymentStrategy strategy;

    @Autowired
    public PaymentService(@Qualifier("alipayStrategy") PaymentStrategy strategy) {
        this.strategy = strategy;
    }
}

// 策略接口
public interface PaymentStrategy {
    void processPayment(BigDecimal amount);
}

// 具体策略实现
@Service("alipayStrategy")
public class AlipayStrategy implements PaymentStrategy { /*...*/ }

@Service("wechatPayStrategy")
public class WechatPayStrategy implements PaymentStrategy { /*...*/ }
十、最佳实践建议
  1. 策略单一职责undefined每个策略类只负责一个具体的算法实现
  2. 无状态策略undefined尽量设计无状态的策略对象,方便复用
  3. 策略文档化undefined为每个策略编写清晰的文档说明使用场景
  4. 性能优化undefined对高频使用的策略考虑对象池技术
  5. 防御式编程undefined在上下文类中进行参数校验:
代码语言:java
复制
public void setStrategy(DiscountStrategy strategy) {
    if (strategy == null) {
        throw new IllegalArgumentException("Strategy cannot be null");
    }
    this.strategy = strategy;
}

扩展示例:文件加密策略

代码语言:java
复制
// 策略接口
public interface EncryptionStrategy {
    String encrypt(String content);
    String decrypt(String encryptedContent);
}

// AES加密策略
public class AesEncryptionStrategy implements EncryptionStrategy {
    // 实现AES加密算法...
}

// RSA加密策略
public class RsaEncryptionStrategy implements EncryptionStrategy {
    // 实现RSA加密算法...
}

// 加密上下文
public class FileEncryptor {
    private EncryptionStrategy strategy;
    
    public FileEncryptor(EncryptionStrategy strategy) {
        this.strategy = strategy;
    }
    
    public void encryptFile(String inputPath, String outputPath) {
        String content = readFile(inputPath);
        String encrypted = strategy.encrypt(content);
        writeFile(outputPath, encrypted);
    }
    
    // 文件读写方法...
}

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 一、模式定义
  • 二、核心角色
  • 三、经典实现(电商促销场景)
  • 四、模式结构UML
  • 五、模式优劣分析
  • 六、应用场景
  • 七、Java标准库应用
  • 八、与相似模式对比
  • 九、高级应用技巧
  • 十、最佳实践建议
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档