昨天在开发的时候遇到一个小问题,本来博主在写Excel的导出,但是Excel所有数据都是动态的,需要根据用户id将用户名显示出来,此时就需要调用用户服务。由于ExcelExportUtil这个类是我自己封装的,用于导出的工具类。里面所有方法都是静态的。这时,如果使用@Autowired注解装配bean,是无法在静态方法中调用非静态成员(方法和变量)的。
这时,我想到以前在跟我司同事协同开发的时候,也遇到过这个问题。那时,我想在activiti工作流的监听器里面调用服务层的方法,也无法直接使用自动装配。然后在我焦头烂额的时候,我同事告诉我,可以使用SpringContextHolder.getBean(xxx.class);这个方法,这是封装好的。那时没有直接去看源码,就为了解决燃眉之急,使用了这个类。碰巧的是,昨天又遇到了相同的问题,所以,在这里记录下来。
SpringContextHolder源码:
/**
* 以静态变量保存Spring ApplicationContext, 可在任何代码任何地方任何时候取出ApplicaitonContext.
*
* @author
* @date 2013-5-29 下午1:25:40
*/
@Service
@Lazy(false)
public class SpringContextHolder implements ApplicationContextAware, DisposableBean {
private static ApplicationContext applicationContext = null;
private static Logger logger = LoggerFactory.getLogger(SpringContextHolder.class);
/**
* 取得存储在静态变量中的ApplicationContext.
*/
public static ApplicationContext getApplicationContext() {
assertContextInjected();
return applicationContext;
}
public static String getRootRealPath() {
String rootRealPath = "";
try {
rootRealPath = getApplicationContext().getResource("").getFile().getAbsolutePath();
} catch (IOException e) {
logger.warn("获取系统根目录失败");
}
return rootRealPath;
}
public static String getResourceRootRealPath() {
String rootRealPath = "";
try {
rootRealPath = new DefaultResourceLoader().getResource("").getFile().getAbsolutePath();
} catch (IOException e) {
logger.warn("获取资源根目录失败");
}
return rootRealPath;
}
/**
* 从静态变量applicationContext中取得Bean, 自动转型为所赋值对象的类型.
*/
@SuppressWarnings("unchecked")
public static <T> T getBean(String name) {
assertContextInjected();
return (T) applicationContext.getBean(name);
}
/**
* 从静态变量applicationContext中取得Bean, 自动转型为所赋值对象的类型.
*/
public static <T> T getBean(Class<T> requiredType) {
assertContextInjected();
return applicationContext.getBean(requiredType);
}
/**
* 清除SpringContextHolder中的ApplicationContext为Null.
*/
public static void clearHolder() {
if (logger.isDebugEnabled()) {
logger.debug("清除SpringContextHolder中的ApplicationContext:" + applicationContext);
}
applicationContext = null;
}
/**
* 实现ApplicationContextAware接口, 注入Context到静态变量中.
*/
@Override
public void setApplicationContext(ApplicationContext applicationContext) {
// logger.debug("注入ApplicationContext到SpringContextHolder:{}",
// applicationContext);
if (SpringContextHolder.applicationContext != null) {
logger.info("SpringContextHolder中的ApplicationContext被覆盖, 原有ApplicationContext为:"
+ SpringContextHolder.applicationContext);
}
SpringContextHolder.applicationContext = applicationContext; // NOSONAR
}
/**
* 实现DisposableBean接口, 在Context关闭时清理静态变量.
*/
@Override
public void destroy() throws Exception {
SpringContextHolder.clearHolder();
}
/**
* 检查ApplicationContext不为空.
*/
private static void assertContextInjected() {
Validate.validState(applicationContext != null,
"applicaitonContext属性未注入, 请在applicationContext.xml中定义SpringContextHolder.");
}
public static ServletContext getServletContext() {
WebApplicationContext wac = ContextLoader.getCurrentWebApplicationContext();
return wac.getServletContext();
}
}
@Service将这个类注入到Spring容器中,不需要再在applicationContext.xml文件定义bean,前提是此类所属的包在Spring扫描的范围内。
@Lazy(false)表示不使用延迟加载bean。
实现ApplicationContextAware这个接口,可以获得ApplicationContext中的所有bean,方便调用Spring容器管理的各个bean。
实现DisposableBean这个接口,当一个bean被销毁时,Spring容器会自动执行这个方法,释放资源。
博主还通过查看其他博客和资料,发现上述代码基本上大同小异。基本上使用上面的代码,就能调用Spring管理的所有bean了。
private static UserService userService = SpringContextHolder.getBean(UserService.class);
这时,就可以成功的调用其他服务中的方法了。
梦想让我成为一名伟人,现实让我变成一个普通人。