策略模式(Strategy Pattern)
指对象有某个行为,但是在不同的场景中,该行为有不同的实现算法。将每个算法封装在独立的类中,使得它们可以互相替换。可以在运行时根据需要选择不同的算法,而不需要修改客户端代码。
主要解决:在有多种算法相似的情况下,使用 if...else 所带来的复杂和难以维护。
关键代码:实现同一个接口。
优点: 1、算法可以自由切换。 2、避免使用多重条件判断。 3、扩展性良好。
缺点: 1、策略类会增多。 2、所有策略类都需要对外暴露。 3、客户端知道所有的算法或行为的情况。
注意事项:如果一个系统的策略多于四个,就需要考虑使用混合模式,解决策略类膨胀的问题。
组成:
1.抽象策略角色: 策略类,通常由一个接口或者抽象类实现。
2.具体策略角色:包装了相关的算法和行为。
3.环境角色:持有一个策略类的引用,最终给客户端调用。环境类可以通过依赖注入、简单工厂等方式来获取具体策略对象。
======================================================== 以上八股文 来源 网络拼凑 ======================================================================
注 : 上述所谓的 算法 不仅仅代表 冒泡排序,归并排序,二分查找 等这种具体的算法, 也代表 某些 业务逻辑,方法运行策略等(不要狭义的看问题)
举个简单小栗子:
当操作数据库时(抽象策略角色), 用到了 增删改查 (具体策略角色)四种 方式, 通过id 查询 , 通过id 修改, 通过 id 删除, 增加一个 id 数据 (维护当前使用 策略的对象【环境角色】)。
/** * 操作数据库时(抽象策略角色) */ public interface OperateStrategy { String operation(String id); } /** * insert操作 (具体策略角色) */ public class OperateInsert implements OperateStrategy{ @Override public String operation(String id) { return "insert into success !! id ==" + id ; } } /** * Select操作 */ public class OperateSelect implements OperateStrategy{ @Override public String operation(String id) { return "select success !! id ==" + id ; } } /** * Delete操作 */ public class OperateDelete implements OperateStrategy{ @Override public String operation(String id) { return "delete success !! id ==" + id ; } } /** * Update操作 */ public class OperateUpdate implements OperateStrategy{ @Override public String operation(String id) { return "update success !! id ==" + id ; } } /** * Context 执行上下文(环境角色) */ public class OperateContext { private OperateStrategy operateStrategy; public OperateContext(OperateStrategy operateStrategy){ this.operateStrategy = operateStrategy; } public String execute(String id){ return operateStrategy.operation(id); } } //执行 public static void main(String[] args) { OperateContext context = new OperateContext(new OperateSelect()); System.out.println(context.execute("select01")); context = new OperateContext(new OperateDelete()); System.out.println(context.execute("delete01")); context = new OperateContext(new OperateUpdate()); System.out.println(context.execute("update01")); context = new OperateContext(new OperateInsert()); System.out.println(context.execute("Insert01")); }
结果:
==========================================收工=================================================================================