Spring 一个接口多个实现,如何根据外部条件来实时替换具体实现类

7 篇文章 0 订阅

代码:

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

作用:

1. 传入一个接口的Class 类型,获取这个class 的所有具体实现,不包括抽象类

2. 还可以将 applicationContext 单独设置一个值,写成一个工具类,结合ApplicationContext 类的其他方法,比如:

getBean(String var1)

需求:

定义了一个接口,来对外提供服务,这个接口下的方法不能随便改变,而接口有一系列实现,且实现还在不断添加,如何在传入外部不同的条件下,实现实时更换接口的实现类

之前的解决办法:

是在controller 中分别引入具体的实现,通过一个外部条件在controller 判断使用那个实现,

弊端:

1.每次增加新的接口,都需要在controller 中在@Autowired 一个依赖,都必须修改Controller 类的代码,

2.之前最多引入的实现类有9个,但是实现还在不断增加,如果继续引入更多类,spring 创建这个controller的时间大大增加,因为引入加载太多类,

修改的办法:

通过spring 的ApplicationContext(应用上下文)的getBeansOfType 方法传入接口类型,一次性获取所有实现类

问题:

1怎么获得实时的应用上下文对象 applicationContext

任何类 实现 ApplicationContextAware 接口,实现setApplicationContext 方法,就会在启动时,向实现类的实现方法注入applicationContext对象

例子:

package com.util;

import com.service.TestService;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.annotation.Configurable;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.stereotype.Component;

import java.util.HashMap;
import java.util.Map;

@Component
public class TestServiceFactory implements ApplicationContextAware {

    private static Map<TypeEnum, TestService> testServiceMap;

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

    public TestService getTestService(TypeEnum typeEnum) {
        return testServiceMap.get(typeEnum);
    }
}

2. 怎么根据外部条件实现获得对应的实现类?

可以在接口中加一个getCode方法,实现类实现这个方法,然后返回一个定义的枚举类型,然后将getBeansOfType获得map进行转换

例子:

package com.util;

public enum  TypeEnum {
    impl1,
    impl2
}

接口

package com.service;

import com.util.TypeEnum;
import org.springframework.stereotype.Service;

@Service
public interface TestService {

    public TypeEnum getCode();

    public String test();

}

实现类

package com.service.impl;

import com.service.TestService;
import com.util.TypeEnum;
import org.springframework.stereotype.Service;

@Service
public class TestServiceImpl1 implements TestService {
    @Override
    public TypeEnum getCode() {
        return TypeEnum.impl1;
    }

    @Override
    public String test() {
        return this.toString();
    }
}
package com.service.impl;

import com.service.TestService;
import com.util.TypeEnum;
import org.springframework.context.annotation.Primary;
import org.springframework.stereotype.Service;

@Service
public class TestServiceImpl2  implements TestService {
    @Override
    public TypeEnum getCode() {
        return TypeEnum.impl2;
    }

    @Override
    public String test() {
        return this.toString();
    }
}

controller类

package com.controller;

import com.util.TestServiceFactory;
import com.util.TypeEnum;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;
import com.service.TestService;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

@Controller
@RequestMapping("test")
public class TestController {

    @Autowired
    TestServiceFactory testServiceFactory;

    private TestService testService;

    @ResponseBody
    @RequestMapping("test")
        public String test(HttpServletRequest request, HttpServletResponse response){
        String type = request.getParameter("type");
        testService = getTestService(type);
        return testService.test();
    }

    public TestService getTestService(String type) {
        TypeEnum typeEnum = null;
        if(type.equals("1")) typeEnum = TypeEnum.impl1;
        if(type.equals("2")) typeEnum = TypeEnum.impl2;
        return testServiceFactory.getTestService(typeEnum);
    }

}

 

  • 4
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
可以的,Spring框架提供了一种叫做"依赖注入"(Dependency Injection)的机制,可以通过配置文件或者注解的方式来实现接口多个实现之间的切换。具体实现方式有以下两种: 1. 通过配置文件实现多个实现之间的切换。在Spring的配置文件,可以使用<bean>元素来配置的实例,其可以指定bean的id和class属性,同时也可以使用<qualifier>元素来指定该bean在多个实现具体实现。 举个例子,假设我们有一个接口UserService,有两个实现UserServiceImpl和UserServiceImpl2。那么可以这样配置: ``` <bean id="userService1" class="com.example.service.impl.UserServiceImpl"/> <bean id="userService2" class="com.example.service.impl.UserServiceImpl2"/> <bean id="userService" class="com.example.service.UserService"> <qualifier value="userService1"/> </bean> ``` 这样,我们就可以在需要使用UserService的地方,通过注入userService来获取具体实现。 2. 通过注解实现多个实现之间的切换。在接口上使用@Qualifier注解来指定具体实现,同时在实现上使用@Primary注解来指定默认的实现。 举个例子,假设我们有一个接口UserService,有两个实现UserServiceImpl和UserServiceImpl2。那么可以这样配置: ``` public interface UserService { void doSomething(); } @Service @Qualifier("userService1") public class UserServiceImpl implements UserService { @Override public void doSomething() { System.out.println("UserServiceImpl do something"); } } @Service @Qualifier("userService2") public class UserServiceImpl2 implements UserService { @Override public void doSomething() { System.out.println("UserServiceImpl2 do something"); } } @Component public class UserController { @Autowired @Qualifier("userService1") private UserService userService; public void test() { userService.doSomething(); } } ``` 在UserController,我们通过@Autowired注解将userService注入进来,同时使用@Qualifier注解来指定具体实现。这样,我们就可以在需要使用UserService的地方,通过注入具体实现来获取不同的实现

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值