多线程并发处理起来通常比较麻烦,如果你使用spring容器来管理业务bean,事情就好办了多了。spring封装了Java的多线程的实现,你只需要关注于并发事物的流程以及一些并发负载量等特性,具体来说如何使用spring来处理并发事务:

1.了解 TaskExecutor接口

Spring的TaskExecutor接口等同于java.util.concurrent.Executor接口。 实际上,它存在的主要原因是为了在使用线程池的时候,将对Java5的依赖抽象出来。 这个接口只有一个方法execute(Runnable task),它根据线程池的语义和配置,来接受一个执行任务。最初创建TaskExecutor是为了在需要时给其他Spring组件提供一个线程池的抽象。 例如ApplicationEventMulticaster组件、JMS的 AbstractMessageListenerContainer和对Quartz的整合都使用了TaskExecutor抽象来提供线程池。 当然,如果你的bean需要线程池行为,你也可以使用这个抽象层。

2. TaskExecutor接口的实现类

(1)SimpleAsyncTaskExecutor 类

这个实现不重用任何线程,或者说它每次调用都启动一个新线程。但是,它还是支持对并发总数设限,当超过线程并发总数限制时,阻塞新的调用,直到有位置被释放。如果你需要真正的池,请继续往下看。

(2)SyncTaskExecutor类

这个实现不会异步执行。相反,每次调用都在发起调用的线程中执行。它的主要用处是在不需要多线程的时候,比如简单的test case。

(3)ConcurrentTaskExecutor 类

这个实现是对Java 5 java.util.concurrent.Executor类的包装。有另一个备选, ThreadPoolTaskExecutor类,它暴露了Executor的配置参数作为bean属性。很少需要使用ConcurrentTaskExecutor, 但是如果ThreadPoolTaskExecutor不敷所需,ConcurrentTaskExecutor是另外一个备选。

(4)SimpleThreadPoolTaskExecutor 类

这个实现实际上是Quartz的SimpleThreadPool类的子类,它会监听Spring的生命周期回调。当你有线程池,需要在Quartz和非Quartz组件中共用时,这是它的典型用处。

(5)ThreadPoolTaskExecutor 类

它不支持任何对java.util.concurrent包的替换或者下行移植。Doug Lea和Dawid Kurzyniec对java.util.concurrent的实现都采用了不同的包结构,导致它们无法正确运行。 这个实现只能在Java 5环境中使用,但是却是这个环境中最常用的。它暴露的bean properties可以用来配置一个java.util.concurrent.ThreadPoolExecutor,把它包装到一个TaskExecutor中。如果你需要更加先进的类,比如ScheduledThreadPoolExecutor,我们建议你使用ConcurrentTaskExecutor来替代。

(6)TimerTaskExecutor类

这个实现使用一个TimerTask作为其背后的实现。它和SyncTaskExecutor的不同在于,方法调用是在一个独立的线程中进行的,虽然在那个线程中是同步的。

(7)WorkManagerTaskExecutor类

这个实现使用了CommonJ WorkManager作为其底层实现,是在Spring context中配置CommonJ WorkManager应用的最重要的类。和SimpleThreadPoolTaskExecutor类似,这个类实现了WorkManager接口,因此可以直接作为WorkManager使用。

3.线程池Demo之 ThreadPoolTaskExecutor

(1)编写测试

  1. import org.springframework.core.task.TaskExecutor;
  2. public class MainExecutor {
  3. private TaskExecutor taskExecutor;
  4. public MainExecutor (TaskExecutor taskExecutor) {
  5. this.taskExecutor = taskExecutor;
  6. }
  7. public void printMessages() {
  8. for(int i = 0; i < 25; i++) {
  9. taskExecutor.execute(new MessagePrinterTask("Message" + i));
  10. }
  11. }
  12. private class MessagePrinterTask implements Runnable {
  13. private String message;
  14. public MessagePrinterTask(String message) {
  15. this.message = message;
  16. }
  17. public void run() {
  18. System.out.println(message);
  19. }
  20. }
  21. }

在业务代码中,通常以for循环的方式执行多个事务
for(int k = 0; k < n; k++) {    
        taskExecutor.execute(new ThreadTransCode());    
    }
其它繁琐的线程管理的事情就交给执行器去管理。
值得注意的事有两点
1, taskExecutor.execute(new ThreadTransCode());  激活的线程都是守护线程,主线程结束,守护线程就会放弃执行,这个在业务中式符合逻辑的,在单元测试中为了看到执行效果,需要自行阻塞主线程。
2, taskExecutor.execute(new ThreadTransCode());    的执行也不是完全安全的,在执行的过程中可能会因为需要的线程查过了线程队列的容量而抛出运行时异常,如有必要需要捕获。

(2)spring的配置

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <!DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN" "/spring-beans.dtd">
  3. <beans>
  4. <!-- 异步线程池 -->
  5. <bean id="threadPool" class="org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor">
  6. <!-- 核心线程数  -->
  7. <property name="corePoolSize" value="10" />
  8. <!-- 最大线程数 -->
  9. <property name="maxPoolSize" value="50" />
  10. <!-- 队列最大长度 >=mainExecutor.maxSize -->
  11. <property name="queueCapacity" value="1000" />
  12. <!-- 线程池维护线程所允许的空闲时间 -->
  13. <property name="keepAliveSeconds" value="300" />
  14. <!-- 线程池对拒绝任务(无线程可用)的处理策略 -->
  15. <property name="rejectedExecutionHandler">
  16. <bean class="java.util.concurrent.ThreadPoolExecutor$CallerRunsPolicy" />
  17. </property>
  18. </bean>
  19. <bean id="mainExecutor" class="supben.MainExecutor">
  20. <property name="threadPool" ref="threadPool" />
  21. </bean>
  22. <bean id="springScheduleExecutorTask" class="org.springframework.scheduling.concurrent.ScheduledExecutorTask">
  23. <property name="runnable" ref="mainExecutor" />
  24. <!-- 容器加载10秒后开始执行 -->
  25. <property name="delay" value="10000" />
  26. <!-- 每次任务间隔 5秒-->
  27. <property name="period" value="5000" />
  28. </bean>
  29. <bean id="springScheduledExecutorFactoryBean" class="org.springframework.scheduling.concurrent.ScheduledExecutorFactoryBean">
  30. <property name="scheduledExecutorTasks">
  31. <list>
  32. <ref bean="springScheduleExecutorTask" />
  33. </list>
  34. </property>
  35. </bean>
  36. </beans>

(3)调用

  1. ApplicationContext appContext = new ClassPathXmlApplicationContext("applicationContext.xml");
  2. MainExecutor te = (MainExecutor)appContext.getBean("taskExecutorExample");
  3. te.printMessages();
  4. System.out.println("11111111111111111111111");

(4)效果

案例:

 @EnableScheduling
@Configuration
public class WebConfig extends SafWebMvcConfigurerAdapter { 、、、、 @Bean
public TaskExecutor taskExecutor() {
ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
executor.setCorePoolSize(5);
executor.setMaxPoolSize(10);
return executor;
} }
  @Autowired
private TaskExecutor taskExecutor; void save2AllTenantSetting(final ForbidScope forbidScope) {
taskExecutor.execute(new Runnable() {
@Override
public void run() {
final List<Tenant> tenants = tenantService.findAll(); for (Tenant tenant : tenants) {
Runnable save2TenantSettingRunnable
= new Save2TenantSettingRunnable(tenant, forbidScope);
taskExecutor.execute(save2TenantSettingRunnable);
}
}
});
}

案例:

首先要将TaskExecutor加入spring容器进行管理

 @Configuration
public class WebMvcConfigurerAdpter extends AbstractWebMvcConfigurerAdpter { @Override
public void configureMessageConverters(List<HttpMessageConverter<?>> converters) {
super.configureMessageConverters(converters);
WafJsonMapper.getMapper().enable(DeserializationFeature.FAIL_ON_NUMBERS_FOR_ENUMS);
} @Bean
public TaskExecutor taskExecutor() {
ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
executor.setCorePoolSize();
executor.setMaxPoolSize();
return executor;
} @Bean
public static PropertySourcesPlaceholderConfigurer propertyConfig() {
return new PropertySourcesPlaceholderConfigurer();
}
}

使用:

 @RestController
@RequestMapping(value = "/v0.1/tasks")
public class TaskController { @Autowired
private TaskService taskService; @RequestMapping()
public Object execute() {
taskService.execute();
Map res = new HashMap();
res.put("result", "success");
return res;
}
}
@Service
public class TaskService { @Autowired
private TaskExecutor executor; public void execute() {
executor.execute(new Runnable() {
@Override
public void run() {
for (int i = 0; i < 10; i++) {
try {
Thread.sleep(1000);
System.out.println("task running ...");
} catch (Exception e) { }
}
}
});
}
}

每次调用/v0.1/tasks 接口时, 不用等到任务执行结束后才会响应,而是响应后,任务还可能在执行 -- 异步而非同步

Spring中线程池的应用的更多相关文章

  1. Spring中线程池的使用

    <bean id="threadPoolTaskExecutor" class="org.springframework.scheduling.concurrent ...

  2. Spring Boot 线程池

    参考 SpringBoot 线程池 程序猿DD-Spring Boot使用@Async实现异步调用:自定义线程池 如何优雅的使用和理解线程池 Spring Boot线程池的使用心得 博客园-Sprin ...

  3. Java中java.util.concurrent包下的4中线程池代码示例

    先来看下ThreadPool的类结构 其中红色框住的是常用的接口和类(图片来自:https://blog.csdn.net/panweiwei1994/article/details/78617117 ...

  4. Spring集成线程池

    自己在程序中手动New很容易造成线程滥用,创建线程也是比较消耗资源的操作,所以建议如果有此需求,将线程池统一交给Spring框架进行管理. 如下: <!--Spring 集成线程池,不允许自己开 ...

  5. spring @Async 线程池使用

    最近公司项目正逐渐从dubbo向springCloud转型,在本次新开发的需求中,全部使用springcloud进行,在使用时线程池,考虑使用spring封装的线程池,现将本次使用心得及内容记录下来 ...

  6. Java中线程池,你真的会用吗?

    在<深入源码分析Java线程池的实现原理>这篇文章中,我们介绍过了Java中线程池的常见用法以及基本原理. 在文中有这样一段描述: 可以通过Executors静态工厂构建线程池,但一般不建 ...

  7. Java并发编程中线程池源码分析及使用

    当Java处理高并发的时候,线程数量特别的多的时候,而且每个线程都是执行很短的时间就结束了,频繁创建线程和销毁线程需要占用很多系统的资源和时间,会降低系统的工作效率. 参考http://www.cnb ...

  8. 沉淀再出发:java中线程池解析

    沉淀再出发:java中线程池解析 一.前言 在多线程执行的环境之中,如果线程执行的时间短但是启动的线程又非常多,线程运转的时间基本上浪费在了创建和销毁上面,因此有没有一种方式能够让一个线程执行完自己的 ...

  9. Java中线程池,你真的会用吗?ExecutorService ThreadPoolExcutor

    原文:https://www.hollischuang.com/archives/2888 在<深入源码分析Java线程池的实现原理>这篇文章中,我们介绍过了Java中线程池的常见用法以及 ...

随机推荐

  1. Jquery 自定义插件写法(示例)

    (function ($) { $.SmsHelper = $.SmsHelper || {}; $.extend($.SmsHelper, { //插件具体实现代码 yzmnum: 60, Ajax ...

  2. Tempdb--Snapshot

    The insert operation does not cause a row version to be generated because there is really no prvious ...

  3. Python 爬虫入门实例(爬取小米应用商店的top应用apk)

    一,爬虫是什么? 爬虫就是获取网络上各种资源,数据的一种工具.具体的可以自行百度. 二,如何写简单爬虫 1,获取网页内容 可以通过 Python(3.x) 自带的 urllib,来实现网页内容的下载. ...

  4. VUE 学习笔记 一 指令

    1.声明式渲染 v-bind 特性被称为指令.指令带有前缀 v-,以表示它们是 Vue 提供的特殊特性 <div id='app'> <span v-bind:title=" ...

  5. ie11下ajax用escape发送中文参数失败

    一个项目中,登录请求是ajax,get模式.登录名无中文可以正常登录:登录名是中文则偶尔可以登录,大部分情况下无法登录,ajax请求无法发送成功. 登录名是用js的escape函数转码. 经过多次测试 ...

  6. unity 移动物体的方式

    1. 简介 在Unity3D中,有多种方式可以改变物体的坐标,实现移动的目的,其本质是每帧修改物体的position. 2. 通过Transform组件移动物体 Transform 组件用于描述物体在 ...

  7. UWP开发砸手机系列(一)—— Accessibility

    因为今天讨论的内容不属于入门系列,所以我把标题都改了.这个啥Accessibility说实话属于及其蛋疼的内容,即如何让视力有障碍的人也能通过声音来使用触屏手机……也许你这辈子也不会接触,但如果有一天 ...

  8. 【Oracle 12c】最新CUUG OCP-071考试题库(58题)

    58.(16-1) choose the best answer: Examine the structure of the BOORSTRANSACTIONS table: Examine the ...

  9. 【Oracle 12c】最新CUUG OCP-071考试题库(57题)

    57.(14-17) choose two: Examine the structure of the DEPARTMENTS table You execute the following comm ...

  10. docker设置引用国内镜像加速

    设置步骤: 1 先到daocloud.io网站注册一个账号 过程略,注册成功后,进入控制台 2 点击控制台上的加速器 拉到中间部分,有一个『主机监控程序』的文字链接,见下图: 然后选择主机类型,我用的 ...