spring注入+策略模式切换业务实现(解决一个接口多个实现动态切换的问题)

一、问题描述:

通过spring托管的service业务类,这个类是通过反射拿到的,经过实验发现这个类只能反射取得service实现了接口的方法,而extends类的方法一律不出现,debug后发现这个servie实例被spring替换成jdkDynmicProxy类,而不是原始对象了,它里面只有service的接口方法,而没有extends 过的super class方法。
结论:当使用策略模式动态切换service实现类时,最好使用原始目标对象。
原因:spring会给我们生成代理对象,有两种方式:JDK动态代理(实现目标类的接口,与目标类是兄弟关系),Cglib动态代理(继承目标类,与目标类是父子关系);当我们的目标类,既实现接口,并且又继承某个类时,就会导致上面的问题出现;

1、获取代理类的目标对象,解决上面遇到的问题:

import java.lang.reflect.Field;  
import org.springframework.aop.framework.AdvisedSupport;  
import org.springframework.aop.framework.AopProxy;  
import org.springframework.aop.support.AopUtils;
/**
 * Description: 能获取JDK动态代理/CGLIB代理对象代理的目标对象。
 * All Rights Reserved.
 * @version 1.0  2015-6-28 上午9:04:32  by zhangbo01@zuche.com创建
 */
public class AopTargetUtils {
     /** 
     * 获取 目标对象 
     * @param proxy 代理对象 
     * @return  
     * @throws Exception 
     */  
    public static Object getTarget(Object proxy) throws Exception {  
          
        if(!AopUtils.isAopProxy(proxy)) {  
            return proxy;//不是代理对象  
        }  
          
        if(AopUtils.isJdkDynamicProxy(proxy)) {  
            return getJdkDynamicProxyTargetObject(proxy);  
        } else { //cglib  
            return getCglibProxyTargetObject(proxy);  
        }  
          
          
          
    }  
  
  
    private static Object getCglibProxyTargetObject(Object proxy) throws Exception {  
        Field h = proxy.getClass().getDeclaredField("CGLIB$CALLBACK_0");  
        h.setAccessible(true);  
        Object dynamicAdvisedInterceptor = h.get(proxy);  
          
        Field advised = dynamicAdvisedInterceptor.getClass().getDeclaredField("advised");  
        advised.setAccessible(true);  
          
        Object target = ((AdvisedSupport)advised.get(dynamicAdvisedInterceptor)).getTargetSource().getTarget();  
          
        return target;  
    }  
  
  
    private static Object getJdkDynamicProxyTargetObject(Object proxy) throws Exception {  
        Field h = proxy.getClass().getSuperclass().getDeclaredField("h");  
        h.setAccessible(true);  
        AopProxy aopProxy = (AopProxy) h.get(proxy);  
          
        Field advised = aopProxy.getClass().getDeclaredField("advised");  
        advised.setAccessible(true);  
          
        Object target = ((AdvisedSupport)advised.get(aopProxy)).getTargetSource().getTarget();  
          
        return target;  
    }  
     
}

二、策略模式动态切换业务实现

/**
 * 
 * @Date: 2020/3/2 13:49
 * @Author: zhenliang.song
 * @Description:
 */
public interface UserService {
    /**
     * 保存用户
     */
    public void saveUser();

}
@Service("userService1")
class UserService1 implements UserService{

    @Override
    public void saveUser() {
        System.out.println("UserService1:业务实现类");
    }
}
@Service("userService2")
class UserService2 implements UserService{

    @Override
    public void saveUser() {
        System.out.println("UserService2:业务实现类");
    }
}

/**
 * 适配策略枚举
 */
enum StrategyCrossEnums{
    STRATEGY_ONE("1", "userService1"),

    STRATEGY_TWO("2", "userService2"),
    ;

    private String code;

    private String className;

    StrategyCrossEnums(String code, String className) {
        this.code = code;
        this.className = className;
    }

    public String getCode() {
        return code;
    }

    public void setCode(String code) {
        this.code = code;
    }

    public String getClassName() {
        return className;
    }

    public void setClassName(String className) {
        this.className = className;
    }
}

/**
 * 管理策略业务:spring启动时加载afterProperTiesSet方法
 */
@Component
class BeanLoaderFactory implements InitializingBean {
    private Map<String, Object> contextMap = new HashMap<>();
    @Override
    public void afterPropertiesSet() throws Exception {
        //获取spring创建的bean
        ApplicationContext applicationContext = SpringContextHolder.getApplicationContext();
        for(StrategyCrossEnums crossEnums : StrategyCrossEnums.values()){
            //获取原始目标对对象
            Object target = AopUtils.getTarget(applicationContext.getBean(crossEnums.getClassName()));
            contextMap.put(crossEnums.getCode(),  target);
        }
    }

    /**
     * 根据code获取业务原始目标对象
     * @param code
     * @return
     */
    public Object getStrategyMap(String code){
        Object target = contextMap.get(code);
        return target;
    }
}
@Controller
@RequestMapping("/strategyController")
class StrategyController{

    @Autowired
    private BeanLoaderFactory beanLoaderFactory;

    /**
     * 调用方法
     * @param code
     */
    @RequestMapping("/method")
    public void method(String code){
        //实现多个业务策略动态切换
        UserService userService = (UserService) beanLoaderFactory.getStrategyMap(code);
        userService.saveUser();
    }
}
  • 0
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 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、付费专栏及课程。

余额充值