Spring通过Bean的name实现策略模式

什么是策略模式?
如果在一个系统里面有许多类,它们之间的区别仅在于它们的行为,那么使用策略模式可以动态地让一个对象在许多行为中选择一种行为。简单理解就是用策略模式替换掉if-else,让代码高内聚低耦合。不同的策略去做不同的事,每个策略之前相互不影响,提高代码的可读性。
策略模式:
优点:
1、算法可以自由切换。
2、避免使用多重条件判断。
3、扩展性良好
缺点:
1、策略类会增多。(我个人觉得不算事缺点吧。人尽其才,物尽其用。)
2、所有策略类都需要对外暴露。(这个才算是缺点,因为我只需要我想要的策略,你把其它我不想要策略都暴露出来了。)
常见的策略模式:Map map = new HashMap<>();这里的Map是策略接口,HashMap是具体实现的策略。Map map = new new TreeMap<>();

场景

假设我不同端的用户和密码都不一样,并且存放到不同的数据库表中,那么按照以前的方法就是通过if-else判断是否是其中的一个端,然后执行代码块。现在我们可以通过策略模式,让前端去调具体端的接口,比如我是web端,那么我前端就传一个web端的具体策略名称参数给后端,后端通过Spring的Bean name获取到具体策略的名称,实现具体的业务,然后把结果返回给前端。这样说可能不太容易理解。

定义策略接口

/**
 * 1. 新增策略接口。
 * 应用场景:多端登录,不同的端使用不同的数据库,所以执行不同的策略。
 */
public interface Strategy<T> {
    //2. 定义策略抽象方法(返回类型是T,让你的接口更加通用化!这里的参数你也可以使用String ... str表示传递多个String类型的参数。但是参数嘛,肯定具体的好点。否则可能就需要强转。)
    T login(String username,String password);
}

实现策略接口

web端

/**
 * 3.实现web登录策略
 */
@Component("webLogin")
public class WebStrategyImpl implements Strategy<WebTokenInfo> {

    @Override
    public WebTokenInfo login(String username, String password) {
        //查询web端的数据库,判断用户名和密码是否正确。

        //正确则返回web端的信息给前端
        WebTokenInfo webTokenInfo = new WebTokenInfo();
        webTokenInfo.setWebToken("web端的token");
        webTokenInfo.setWebUserId(1);
        return webTokenInfo;
    }
}

MiniApps端

/**
 * 3.实现小程序登录策略
 */
@Component("miniAppsLogin")
public class MiniAppsStrategyImpl implements Strategy<MiniAppsTokenInfo> {

    @Override
    public MiniAppsTokenInfo login(String username, String password) {
        //查询web端的数据库,判断用户名和密码是否正确。

        //正确则返回web端的信息给前端
        MiniAppsTokenInfo miniAppsTokenInfo = new MiniAppsTokenInfo();
        miniAppsTokenInfo.setMiniAppsToken("小程序端的token");
        miniAppsTokenInfo.setMiniAppsUserId(2);
        return miniAppsTokenInfo;
    }
}

根据Spring的Bean name去获取前端传递过来的具体策略参数调用具体的策略

/**
 * 4.根据前端传的bean的名字去执行具体的策略
 */
@Component
public class ExecuteStrategy {
	//strategy就是@Component("name")的name,返回的是个Object更加通用,因为我们无需关注返回对象是什么,早在实现具体的策略的时候我们以及定义好了返回的对象,而Object是所有对象的基类(父类),需要关注Object具体是什么的只有前端!
    public Object executeStrategy(String strategy, String username, String password) {
        return  ((Strategy<?>) ApplicationContextUtil.getBean(strategy)).login(username, password);
    }
}

Controller接收前端传递的策略参数以及登录接口参数

/**
 * 5.策略模式登录接口,实现多端登录
 */
@RestController
@RequestMapping("strategy")
public class StrategyController {
    @Autowired
    private ExecuteStrategy executeStrategy;

    /**
     * 登录接口
     * @param strategy 策略名称,即:XXXStrategyImpl的@Component("name")的name。
     * @param username 用户名(这里是为了策略而策略,所以假设用户名不同端存在不同的数据库中)
     * @param password 用户密码(这里是为了策略而策略,所以假设密码不同端存在不同的数据库中)
     */
    @GetMapping("{strategy}")//这样写就可以让前端以为调不同的接口,实际上在后端是同一个接口,调的是impl的bean name实现类
    public Object login(@PathVariable String strategy, @RequestParam("username") String username,@RequestParam("password") String password){
        //根据前端传递的strategy策略参数,通过Spring的Bean的name去执行具体的策略。
        return this.executeStrategy.executeStrategy(strategy,username,password);
    }
}

其它

ApplicationContextUtil工具类

/**
 * 获取Bean
 */
@Component
public class ApplicationContextUtil implements ApplicationContextAware {
    //构造函数私有化,防止其它人实例化该对象
    private ApplicationContextUtil(){}
    private static ApplicationContext applicationContext;

    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        ApplicationContextUtil.applicationContext = applicationContext;
    }

    //获取applicationContext(一般情况下用户数。因为我们暂时只需要获取bean就不需要获取其它内容了,而获取bean直接调用下面的方法即可。)
    public static ApplicationContext getApplicationContext() {
        return applicationContext;
    }

    //通过name获取 Bean.(推荐,因为bean的name是唯一的,出现重名的bean启动会报错。)
    public static Object getBean(String name) {
        return getApplicationContext().getBean(name);
    }

    //通过class获取Bean.(确保bean的name不会重复。因为可能会出现在不同包的同名bean导致获取到2个实例)
    public static <T> T getBean(Class<T> clazz) {
        return getApplicationContext().getBean(clazz);
    }

    //通过name,以及Clazz返回指定的Bean(这个是最稳妥的)
    public static <T> T getBean(String name, Class<T> clazz) {
        return getApplicationContext().getBean(name, clazz);
    }

}

MiniAppsTokenInfo小程序返回vo,这个就是上面说的,由于策略接口返回的是个泛型T,我们已经在具体的策略里面定义好了返回类即可。

/**
 * web端token存储的信息
 */
@Data
public class MiniAppsTokenInfo {
    private String miniAppsToken; //小程序端的token
    private Integer miniAppsUserId; //小程序端的用户id
}

WebTokenInfo web端返回vo(同上)

/**
 * web端token存储的信息
 */
@Data
public class WebTokenInfo {
    private String webToken; //web端的token
    private Integer webUserId; //web端的用户id
}

模拟前端调用策略

调用web端的登录策略
在这里插入图片描述
调用小程序端的登录策略
在这里插入图片描述
调用没有的策略,就会抛出异常。
在这里插入图片描述
这里只是模拟,不要纠结为什么在url穿用户名和密码。

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
面试题包括以下十九部分:Java 基础、容器、多线程、反射、对象拷贝、Java Web 模块、异常、网络、设计模式、Spring/Spring MVC、Spring Boot/Spring Cloud、Hibernate、Mybatis、RabbitMQ、Kafka、Zookeeper、MySql、Redis、JVM 。 目录: 一、Java 基础 1.JDK 和 JRE 有什么区别? 2.== 和 equals 的区别是什么? 3.两个对象的 hashCode()相同,则 equals()也一定为 true,对吗? 4.final 在 java 中有什么作用? 5.java 中的 Math.round(-1.5) 等于多少? 6.String 属于基础的数据类型吗? 7.java 中操作字符串都有哪些类?它们之间有什么区别? 8.String str="i"与 String str=new String(“i”)一样吗? 9.如何将字符串反转? 10.String 类的常用方法都有那些? 11.抽象类必须要有抽象方法吗? 12.普通类和抽象类有哪些区别? 13.抽象类能使用 final 修饰吗? 14.接口和抽象类有什么区别? 15.java 中 IO 流分为几种? 16.BIO、NIO、AIO 有什么区别? 17.Files的常用方法都有哪些? 二、容器 18.java 容器都有哪些? 19.Collection 和 Collections 有什么区别? 20.List、Set、Map 之间的区别是什么? 21.HashMap 和 Hashtable 有什么区别? 22.如何决定使用 HashMap 还是 TreeMap? 23.说一下 HashMap 的实现原理? 24.说一下 HashSet 的实现原理? 25.ArrayList 和 LinkedList 的区别是什么? 26.如何实现数组和 List 之间的转换? 27.ArrayList 和 Vector 的区别是什么? 28.Array 和 ArrayList 有何区别? 29.在 Queue 中 poll()和 remove()有什么区别? 30.哪些集合类是线程安全的? 31.迭代器 Iterator 是什么? 32.Iterator 怎么使用?有什么特点? 33.Iterator 和 ListIterator 有什么区别? 34.怎么确保一个集合不能被修改? 三、多线程 35.并行和并发有什么区别? 36.线程和进程的区别? 37.守护线程是什么? 38.创建线程有哪几种方式? 39.说一下 runnable 和 callable 有什么区别? 40.线程有哪些状态? 41.sleep() 和 wait() 有什么区别? 42.notify()和 notifyAll()有什么区别? 43.线程的 run()和 start()有什么区别? 44.创建线程池有哪几种方式? 45.线程池都有哪些状态? 46.线程池中 submit()和 execute()方法有什么区别? 47.在 java 程序中怎么保证多线程的运行安全? 48.多线程锁的升级原理是什么? 49.什么是死锁? 50.怎么防止死锁? 51.ThreadLocal 是什么?有哪些使用场景? 52.说一下 synchronized 底层实现原理? 53.synchronized 和 volatile 的区别是什么? 54.synchronized 和 Lock 有什么区别? 55.synchronized 和 ReentrantLock 区别是什么? 56.说一下 atomic 的原理? 四、反射 57.什么是反射? 58.什么是 java 序列化?什么情况下需要序列化? 59.动态代理是什么?有哪些应用? 60.怎么实现动态代理? 五、对象拷贝 61.为什么要使用克隆? 62.如何实现对象克隆? 63.深拷贝和浅拷贝区别是什么? 六、Java Web 64.jsp 和 servlet 有什么区别? 65.jsp 有哪些内置对象?作用分别是什么? 66.说一下 jsp 的 4 种作用域? 67.session 和 cookie 有什么区别? 68.说一下 session 的工作原理? 69.如果客户端禁止 cookie 能实现 session 还能用吗? 70.spring mvc 和 struts 的区别是什么? 71.如何避免 sql 注入? 72.什么是 XSS 攻击,如何避免? 73.什么是 CSRF 攻击,如何避免? 七、异常 74.throw 和 throws 的区别? 75.final、finally、finalize 有什么区别? 76.try-catch-finally 中哪个部分

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值