使用Spring特性实现接口多实现类的动态调用

背景

org.springframework.beans及org.springframework.context这两个包是Spring IoC容器的基础,
其中重要的类有BeanFactory,BeanFactory是IoC容器的核心接口,其职责包括:实例化、定位、配置应用程序中的
对象及建立这些对象间的依赖关系。

ApplicationContext作为BeanFactory的子类,在Bean管理的功能上得到了很大的增强,也更易于与Spring AOP集成使用。
今天我们要讨论的并不是BeanFactory或者ApplicationContext的实现原理,而是对ApplicationContext的一种实际应用方式。

问题的提出

在实际工作中,我们经常会遇到一个接口及多个实现类的情况,并且在不同的条件下会使用不同的实现类。从使用方式上看,有些类似SPI的用法,
但是由于SPI的使用并不是太方便,那么怎么办呢?我们可以借助ApplicationContext的getBeansOfType来实现我们需要的结果。

首先我们看一下这个方法的签名

<T> Map<String, T> getBeansOfType(Class<T> type) throws BeansException;

从上面的代码上我们可以看出来这个方法能返回一个接口的全部实现类(前提是所有实现类都必须由Spring IoC容器管理)。

接下来看看我们遇到的问题是什么?

“假设从A点到B点有多种交通方式,每种交通方式的费用不同,可以根据乘客的需要进行选择”(好吧,我承认这是个非常蹩脚的需求,
但是可以联想一下类似的需求,比如支付方式、快递公司,在页面提供几个选项,业务代码根据选项的不同选择不同的实现类实例进行调用)。

实现

回到我们的例子,按照这个交通方式的需求,我们的设计如下:有一个交通方式的接口,接口有两个方式,一个查询费用、一个查询该交通方式的类型,同时,我们可以用一个枚举类型类标识交通类型。

我们还需要一个工厂类来根据交通类型标识查找该交通类型的Bean实例,从而使用该实例,获得交通类型的详细信息及该交通类型的操作。

在这里插入图片描述

代码如下:

接口:

/**
 * 交通方式
 */
public interface TrafficMode {

    /**
     * 查询交通方式编码
     * @return 编码
     */
    TrafficCode getCode();

    /**
     * 查询交通方式的费用,单位:分
     * @return 费用
     */
    Integer getFee();
    
}

枚举:

/**
 * 交通类型枚举
 */
public enum TrafficCode {

    TRAIN,
    BUS
    
}

接口有两个实现类:

/**
 * 汽车方式
 */
@Component
public class BusMode implements TrafficMode {

    @Override
    public TrafficCode getCode() {
        return TrafficCode.BUS;
    }

    @Override
    public Integer getFee() {
        return 10000;
    }
    
}
/**
 * 火车方式
 */
@Component
public class TrainMode implements TrafficMode {

    @Override
    public TrafficCode getCode() {
        return TrafficCode.TRAIN;
    }

    @Override
    public Integer getFee() {
        return 9000;
    }
    
}

工厂类:

/**
 * 交通方式工厂类
 */
@Component
public class TrafficModeFactory implements ApplicationContextAware {

    private static Map<TrafficCode, TrafficMode> trafficBeanMap;

    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        Map<String, TrafficMode> map = applicationContext.getBeansOfType(TrafficMode.class);
        trafficBeanMap = new HashMap<>();
        map.forEach((key, value) -> trafficBeanMap.put(value.getCode(), value));
    }

    public static <T extends TrafficMode> T getTrafficMode(TrafficCode code) {
        return (T)trafficBeanMap.get(code);
    }

}

验证

有了上面的代码之后,我们一起通过单元测试来看一下效果,单元测试代码片段如下:

@Test
public void testGetTrafficMode() {
    TrafficMode mode = TrafficModeFactory.getTrafficMode(TrafficCode.BUS);
    Assert.assertEquals(mode.getFee().intValue(), 10000);

    mode = TrafficModeFactory.getTrafficMode(TrafficCode.TRAIN);
    Assert.assertEquals(mode.getFee().intValue(), 9000);
}

运行之后的结果呢?必然是通过。

关于SPI

文章到这里,有同学可能会问:这和SPI有什么区别呢?SPI同样也能实现同样的功能啊。首先说明一下,SPI是JDK自带的功能,虽然历史是比较久远的了,但是不代表它不好,而且有些场景非SPI不可(比如JDBC)。

我们明确一下SPI是什么以及它的设计是用来做什么的。SPI的全名为Service Provider Interface(服务提供接口),因为这个是针对厂商或者插件的。比较经典的用法就是JDBC,java提供了标准的JDBC接口,每个数据库厂商提供自己的数据库驱动实现。下面是JDBC驱动的接口:

public interface Driver {
    Connection connect(String var1, Properties var2) throws SQLException;

    boolean acceptsURL(String var1) throws SQLException;

    DriverPropertyInfo[] getPropertyInfo(String var1, Properties var2) throws SQLException;

    int getMajorVersion();

    int getMinorVersion();

    boolean jdbcCompliant();

    Logger getParentLogger() throws SQLFeatureNotSupportedException;
}

下面是MySQL的JDBC驱动实现的SPI配置:

在这里插入图片描述

关于SPI我们点到为止,这里只是要说明SPI和我们前面例子中使用的AP具有不同的适用场景。

在前面的例子里,我们用的是什么呢?Spring的API,API的全称为Application Programming Interface(应用程序编程接口),在一个应用内部,使用API是非常便捷的方式,也是最直接的方式。

总结一下,就是编程中,我们使用API是最多的。当然我们也会使用SPI(虽然我们不是严格意义上的厂商或者插件),使用SPI也是在特定场景下为了解决问题的一种途径。

关于SPI更多的内容,后续会有文章专门介绍。

【附】关于SPI的约定:当服务的提供者,提供了服务接口的一种实现之后,在jar包的META-INF/services/目录里同时创建一个以服务接口命名的文件。该文件里就是实现该服务接口的具体实现类。而当外部程序装配这个模块的时候,就能通过该jar包META-INF/services/里的配置文件找到具体的实现类名,并装载实例化,完成模块的注入。 基于这样一个约定就能很好的找到服务接口的实现类,而不需要再代码里制定。jdk提供服务实现查找的一个工具类:java.util.ServiceLoader,通过其load方法,传入接口便能获得其实现类。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: SpringSecurity 提供了一种功能强大的机制来实现登录功能,它可以使用 LDAP,数据库,内存等不同的认证源来验证用户的身份。它还支持许多安全特性,如认证失败的重试,多因素认证,强制密码更改,密码恢复,会话管理以及安全响应等。 ### 回答2: 使用Spring Security实现登录功能主要分为以下几个步骤: 1. 引入Spring Security依赖:在项目的`pom.xml`文件中添加Spring Security相关依赖,如`spring-boot-starter-security`。 2. 配置Spring Security:创建一个配置类,如`SecurityConfig`,并标注`@EnableWebSecurity`注解,该注解将开启Spring Security的功能。在配置类中可以通过重写`configure(HttpSecurity http)`方法来配置登录所需的认证规则、URL路径、页面等。 3. 创建用户服务类:创建一个类,如`UserDetailsServiceImpl`,实现`UserDetailsService`接口,并重写其中的`loadUserByUsername(String username)`方法,该方法用于根据用户名从数据库或其他存储方式中获取用户信息。 4. 自定义登录页面:在Spring Security的配置类中配置自定义的登录页面,可以通过重写`configure(HttpSecurity http)`方法中的`loginPage(String loginPage)`指定登录页面的URL路径,并自定义实现该页面。 5. 处理用户认证:在用户登录的时候,Spring Security将会自动调用`UserDetailsService`中的`loadUserByUsername(String username)`方法加载用户信息,并进行认证,验证用户名和密码是否正确。 6. 处理用户授权:需要根据用户的角色或权限来限制用户可以访问的页面或功能,可以通过配置类的`configure(HttpSecurity http)`方法中的`authorizeRequests()`来实现。 7. 添加登录成功和失败的处理器:可以通过自定义实现`AuthenticationSuccessHandler`接口和`AuthenticationFailureHandler`接口来处理登录成功和失败的逻辑。 以上就是使用Spring Security实现登录功能的基本步骤。具体的实现需要根据实际项目的需求进行配置和定制。 ### 回答3: 要使用Spring Security实现登录功能,可以按照以下步骤进行操作: 1. 添加依赖:在项目的pom.xml文件中添加Spring Security的依赖。 2. 配置Spring Security:创建一个继承自WebSecurityConfigurerAdapter的配置类,并使用@EnableWebSecurity注解来启用Spring Security。在配置类中,可以配置认证提供者、用户权限、请求授权等。 3. 添加登录页面:创建一个登录页面,并在登录页面中提供用户名和密码的输入框和登录按钮。 4. 配置认证:在配置类中,重写configure方法,通过AuthenticationManagerBuilder的userDetailsService方法来配置用户认证信息。可以选择使用内存、数据库或其他方式来存储用户信息。 5. 配置登录处理:在配置类中,重写configure方法,通过HttpSecurity的formLogin方法配置登录处理。可以指定登录页面的URL、登录成功和失败的处理URL等。 6. 添加用户权限:如果需要进行请求授权,可以在配置类中通过重写configure方法,使用HttpSecurity的authorizeRequests方法配置请求的权限。可以根据URL、角色或其他条件进行授权。 7. 运行项目:启动项目后,访问登录页面,输入正确的用户名和密码,然后点击登录按钮进行登录。如果登录成功,将根据配置的页面跳转规则进行跳转。 使用Spring Security可以实现简单而强大的登录功能,并且可以根据需求进行灵活的配置和扩展。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值