今天项目中根据返回结果集中的地址我要调用webservice接口,你知道调用接口要花费很多时间,我不可能等接口调用完成后我再返回给用户响应。所以在响应之前我启用一个线程去调用。
那为什么要用线程池:
线程池作用就是限制系统中执行线程的数量。
根据系统的环境情况,可以自动或手动设置线程数量,达到运行的最佳效果;少了浪费了系统资源,多了造成系统拥挤效率不高。用线程池控制线程数量,其他线程排队等候。一个任务执行完毕,再从队列的中取最前面的任务开始执行。若队列中没有等待进程,线程池的这一资源处于等待。当一个新任务需要运行时,如果线程池中有等待的工作线程,就可以开始运行了;否则进入等待队列。
为什么要用线程池:
- 减少了创建和销毁线程的次数,每个工作线程都可以被重复利用,可执行多个任务
- 可以根据系统的承受能力,调整线程池中工作线线程的数目,防止因为因为消耗过多的内存,而把服务器累趴下(每个线程需要大约1MB内存,线程开的越多,消耗的内存也就越大,最后死机)
实现如下:
Model.Manager inner = this.new Manager(logThread);
Thread thread = new Thread(inner);
/**
* 启动一个新线程处理日志记录,日志记录采用异步方式进行保存,后期改用线程池实现
*/
this.getSystemModel().setRequestEndDate(System.currentTimeMillis());
thread.start();
Manager其实是个线程,里面的实现:
/**
* 线程池的调用线程
*/
class Manager implements Runnable
{
private LogThread thread;
public Manager(LogThread thread)
{
this.thread = thread;
}
public void run()
{
threadPool.execute(thread);
}
}
*/
class Manager implements Runnable
{
private LogThread thread;
public Manager(LogThread thread)
{
this.thread = thread;
}
public void run()
{
threadPool.execute(thread);
}
}
threadPool为线程池,我使用的jdk自带的线程池的创建方法:
private static final ExecutorService threadPool = Executors.newFixedThreadPool(5);
但是还有一个spring 自带的线程池:
private ThreadPoolTaskExecutor threadPoolTaskExecutor;
提供set方法如下:
public void setThreadPoolTaskExecutor(
ThreadPoolTaskExecutor threadPoolTaskExecutor)
{
this.threadPoolTaskExecutor = threadPoolTaskExecutor;
}
ThreadPoolTaskExecutor threadPoolTaskExecutor)
{
this.threadPoolTaskExecutor = threadPoolTaskExecutor;
}
配置xml如下:
<bean id ="taskExecutor" class ="org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor" >
<!-- 线程池维护线程的最少数量 -->
<property name="corePoolSize" value ="5" />
<!-- 线程池维护线程所允许的空闲时间 -->
<property name ="keepAliveSeconds" value ="30000" />
<!-- 线程池维护线程的最大数量 -->
<property name ="maxPoolSize" value ="1000" />
<!-- 线程池所使用的缓冲队列 -->
<property name ="queueCapacity" value ="200" />
</bean>
<!-- 线程池维护线程的最少数量 -->
<property name="corePoolSize" value ="5" />
<!-- 线程池维护线程所允许的空闲时间 -->
<property name ="keepAliveSeconds" value ="30000" />
<!-- 线程池维护线程的最大数量 -->
<property name ="maxPoolSize" value ="1000" />
<!-- 线程池所使用的缓冲队列 -->
<property name ="queueCapacity" value ="200" />
</bean>