我有以下配置:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:task="http://www.springframework.org/schema/task"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-3.0.xsd
http://www.springframework.org/schema/task http://www.springframework.org/schema/task/spring-task-3.0.xsd">
<context:component-scan base-package="com.abc" />
<task:annotation-driven executor="executor"/>
<task:executor id="executor" pool-size="2"/>
</beans>
然后是以下课程
public class SomeClassImpl implements SomeClass {
@Async
@Override // overridden from the interface
public void doSomething(){
System.out.println("doing something async");
}
}
测试:
@ContextConfiguration("classpath:test-config.xml") // with the xml config above
@RunWith(value = SpringJUnit4ClassRunner.class)
public class SomeClassTest {
@Autowired
private SomeClass someClass;
@Test
public void testSomething() throws Exception {
System.out.println("Calling doSomething");
someClass.doSomething();
Thread.sleep(5000);
}
}
当我运行测试时,一切都按预期工作。但是当我调用someClass.doSomething()并且我注意到以下内容时,我附加了调试器以逐步执行实际发生的事情:
为什么SimpleAsyncTaskExecutor正在创建4个线程?我知道如果我从task:annotation-driven xml元素中删除executor属性,AsyncExecutionInterceptor将使用SimpleAsyncTaskExecutor。但是因为我已经声明了一个任务执行器并从注释驱动元素中引用它,为什么要创建一个SimpleAsyncTaskExecutor呢?
答案 0 :(得分:6)
首先,如果使用Java 5及更高版本(Java Futures等所需),默认的TaskExecutor实现应该是ThreadPoolTaskExecutor,它与您的调试跟踪输出相匹配。这是实际执行测试代码的线程管理器。
SimpleAsyncTaskExecutor可能是作为Spring任务执行器框架的一部分启动的,可能来自测试上下文文件中的另一个注释。 Spring容器可能正在启动SimpleAsyncTaskExecutor类的四个实例。根据Spring文档,此版本的TaskExecutor永远不会重新使用线程来满足新请求(它将启动一个新线程):
SimpleAsyncTaskExecutor
此实现不会重用任何线程,而是启动新线程 每次调用。但是,它确实支持并发限制,这将限制任何 超出限制的调用,直到插槽被释放。如果你正在寻找 对于真正的池,请继续向下滚动页面。
因此,这可能是测试上下文与Spring容器之间交互的结果。
我相信在这个场景中你有一个运行测试代码的线程,这是你期望的,基于单个请求。 Spring TaskExecutor实现利用一个名为ConcurrencyThrottleSupport的辅助类,它应该限制(限制)执行中的并发线程数。在您的情况下,这应该是2,如pool-size属性所示。但是,要运行这一个测试,它应该永远不需要分配额外的线程,并且跟踪输出与此一致。