《设计模式》之策略模式

策略模式定义

比如对象的某个行为,在不同场景有不同实现方式,可以将这些行为的具体实现定义为一组策略,每个实现类实现种策略,在不同场景使用不同的实现,并且可以自由切换策略。

策略模式结构

策略模式需要一个策略接口,不同的策略实现不同的实现类,在具体业务环境中仅持有该策略接口,根据不同的场景使用不同的实现类即可。

面向接口编程,而不是面向实现。

优点

1、干掉繁琐的 if、switch 判断逻辑;

2、代码优雅、可复用、可读性好;

3、符合开闭原则(对修改关闭, 对扩展开放),扩展性好、便于维护;

缺点

1、策略如果很多的话,会造成策略类膨胀;

2、使用者必须清楚所有的策略类及其用途;

策略模式代码示例

  • 基础登录接口

    //基础登录接口
    public interface BaseLoginService {
    void login(BaseLoginContext context);
    }

  • 策略上下文实现

    @Data
    public class BaseLoginContext {
    private String userName;

      private String password;
    
      private BaseLoginService baseLoginService;
    
      public BaseLoginContext(String userName, String password, BaseLoginService baseLoginService) {
          this.userName = userName;
          this.password = password;
          this.baseLoginService = baseLoginService;
      }
    
      public void login(){
          baseLoginService.login(this);
      }
    

    }

  • 定义具体的策略实现类

  1. 实现类1

    public class SimpleLoginServiceImpl implements BaseLoginService {
    @Override
    public void login(BaseLoginContext context) {
    System.out.println("简单登录,当前登录的人:" + context.getUserName() + ",密码:" + context.getPassword());
    }
    }

  2. 实现类2

    public class HardLoginServiceImpl implements BaseLoginService {
    @Override
    public void login(BaseLoginContext context) {
    System.out.println("复杂登录,当前登录的人:" + context.getUserName() + ",密码:" + context.getPassword());
    }
    }

  • 调用

    public static void main(String[] args) {
    BaseLoginService simple = new SimpleLoginServiceImpl();
    BaseLoginContext simpleContext = new BaseLoginContext("simple","simple",simple);
    simpleContext.login();

          BaseLoginService hard = new HardLoginServiceImpl();
          BaseLoginContext hardContext = new BaseLoginContext("hard","hard",hard);
          hardContext.login();
      }
    
  • 结果

扩展策略

方式1

在策略的算法实现上添加自己需要的数据的方式

public class CommonLoginServiceImpl implements BaseLoginService {

    private String userId;

    public CommonLoginServiceImpl(String userId) {
        this.userId = userId;
    }

    public String getUserId() {
        return userId;
    }

    @Override
    public void login(BaseLoginContext context) {
        System.out.println("复杂登录,当前登录的人:" + context.getUserName() + ",账号:" + getUserId() + ",密码:" + context.getPassword());
    }
}
  • 调用

    public static void main(String[] args) {
    BaseLoginService simple = new SimpleLoginServiceImpl();
    BaseLoginContext simpleContext = new BaseLoginContext("simple","simple",simple);
    simpleContext.login();

          BaseLoginService hard = new HardLoginServiceImpl();
          BaseLoginContext hardContext = new BaseLoginContext("hard","hard",hard);
          hardContext.login();
    
          BaseLoginService common = new CommonLoginServiceImpl("001");
          BaseLoginContext commonContext = new BaseLoginContext("common","common",common);
          commonContext.login();
      }
    
  • 结果

方式2

扩展上下文的方式

public class NewLoginContext extends BaseLoginContext{
    private String userId;

    public String getUserId() {
        return userId;
    }

    public void setUserId(String userId) {
        this.userId = userId;
    }

    public NewLoginContext(String userName, String password, String userId, BaseLoginService baseLoginService) {
        super(userName, password, baseLoginService);
        this.userId = userId;
    }
}
  • 实现类

    public class CommonLoginServiceImpl implements BaseLoginService {

      private String userId;
    
      public CommonLoginServiceImpl(String userId) {
          this.userId = userId;
      }
    
      public String getUserId() {
          return userId;
      }
    
      @Override
      public void login(BaseLoginContext context) {
          NewLoginContext newLoginContext = (NewLoginContext)context;
    

    // System.out.println("common1登录,当前登录的人:" + context.getUserName() + ",账号:" + getUserId() + ",密码:" + context.getPassword());
    System.out.println("common2登录,当前登录的人:" + context.getUserName() + ",账号:" + newLoginContext.getUserId() + ",密码:" + context.getPassword());
    }
    }

  • 调用

    public static void main(String[] args) {
    BaseLoginService simple = new SimpleLoginServiceImpl();
    BaseLoginContext simpleContext = new BaseLoginContext("simple","simple",simple);
    simpleContext.login();

          BaseLoginService hard = new HardLoginServiceImpl();
          BaseLoginContext hardContext = new BaseLoginContext("hard","hard",hard);
          hardContext.login();
    

    // BaseLoginService common1 = new CommonLoginServiceImpl("001");
    // BaseLoginContext commonContext1 = new BaseLoginContext("common","common",common1);
    // commonContext1.login();

          BaseLoginService common2 = new CommonLoginServiceImpl("001");
          NewLoginContext commonContext2 = new NewLoginContext("common","common","002",common2);
          commonContext2.login();
      }
    
  • 结果

相关推荐
笃励18 分钟前
Java面试题二
java·开发语言·python
易雪寒36 分钟前
IDEA在git提交时添加忽略文件
java·git·intellij-idea
打码人的日常分享1 小时前
企业人力资源管理,人事档案管理,绩效考核,五险一金,招聘培训,薪酬管理一体化管理系统(源码)
java·数据库·python·需求分析·规格说明书
27669582921 小时前
京东e卡滑块 分析
java·javascript·python·node.js·go·滑块·京东
爱写代码的刚子1 小时前
C++知识总结
java·开发语言·c++
冷琴19961 小时前
基于java+springboot的酒店预定网站、酒店客房管理系统
java·开发语言·spring boot
daiyang123...2 小时前
IT 行业的就业情况
java
拉里小猪的迷弟2 小时前
设计模式-结构型-常用:代理模式、桥接模式、装饰者模式、适配器模式
设计模式·代理模式·桥接模式·适配器模式·装饰器模式
爬山算法2 小时前
Maven(6)如何使用Maven进行项目构建?
java·maven
.生产的驴2 小时前
Electron Vue框架环境搭建 Vue3环境搭建
java·前端·vue.js·spring boot·后端·electron·ecmascript