前言
刚刚接触到这个领域的时候,我感觉无从下手。一边查资料一边写代码看运行日志,碰到了一些坑,也学到了很多知识—不仅仅是并发编程方面的,也包括spring、IoC和bean的知识。在这里做个记录吧!
正文
需求
一般情况下,在Spring的项目中很少有使用多线程处理任务的,没错,大多数时候我们都是使用Spring MVC开发的web项目,默认的Controller,Service,Dao组件的作用域都是单实例,无状态,然后被并发多线程调用。现在,我们的项目是一个服务平台项目,对外提供大量的服务api让用户调用。在我们的项目里还用到了服务路由(准备再写一篇博客讲这个),什么是服务路由呢,就是我们真正供用户调用的服务api实际上都是第三方提供的服务api,只不过用的url是我们的url。现在,我们的需求是:在项目运行的时候定期检测这些第三方服务是否可用,并提供服务质量的报告,如果第三方服务不可用,在我们服务平台上将对应的api停掉。很自然地想到,这样一个监测功能应该在后台运行,不能和项目原先的功能冲突。因此,我们要在启动类的main方法里开辟新的线程去执行监控的这个功能。
考虑另外一个事情,如何在启动类里面去执行某个controller的某个方法呢?一般来说,执行controller内的方法都是在外部访问这个方法的url。那么,在启动类的main方法里如何访问这个controller呢?说起来其实这是个挺简单的事情:把controller当做一个普通的类即可,如果是访问某个static方法,问题变得很简单,直接在main方法里调用这个controller类名.方法即可。如果不是static方法(通常都不是static方法),则需要controller的实例,也就是这个bean,直接new出来的实例是没法使用的,即使可以new成功,bean里面依赖的其他组件比如dao,是没法初始化的。这是因为绕过了spring,默认的spring初始化一个类的时候,其相关依赖的组件都会被初始化,而自己new出来的实例,不具备这种功能。你或许会想,可以使用依赖注入的方法,利用注解@Autowired将controller实例注入,实际上这是不可行的,因为main方法是个static方法,不可以使用非static变量,而如果使用static修饰controller变量,那么是不能注入的,这是因为spring容器管理的都是对象的属性,而static变量是属于类的,具体可见其他博客。
那到底该怎么办?我们还是要在启动类的main方法里获取controller的bean,然后去执行controller实例里面的方法。项目启动之后,项目里配置的所有bean都被spring管理,我们需要手动地获取ApplicationContext,然后再利用getBean方法去手动地获得bean。
手动获取bean
手动获取bean的方法很多,我这里不多说。我们采取的是其中一种,即实现ApplicationContextAware接口,让这个实现类作为一个bean被spring管理,getBean为static方法,可以直接在main方法里调用,即可通过bean的名称获取bean,代码如下:
@Component
public class ApplicationContextProvider implements ApplicationContextAware {
private static ApplicationContext context;
private ApplicationContextProvider(){
}
@Override
public void setApplicationContext(ApplicationContext applicationContext)throws BeansException{
context=applicationContext;
}
public static <T> T getBean(String name,Class<T> aClass){
return context.getBean(name,aClass);
}
public static ApplicationContext getContext(){
return context;
}
}
启动类main方法里获取bean的方法是:
MonitorTask m1= ApplicationContextProvider.getBean("mTask",MonitorTask.class)