【发布时间】:2013-05-16 19:34:57
【问题描述】:
我需要在 Java 中构建一个工人池,其中每个工人都有自己的连接套接字;当工作线程运行时,它使用套接字但保持打开以供以后重用。我们之所以决定采用这种方法,是因为与在 ad-hoc 基础上创建、连接和销毁套接字相关的开销需要太多开销,因此我们需要一种方法,通过该方法,工作池可以通过其套接字连接进行预初始化,准备好在保持套接字资源不受其他线程影响的同时继续工作(套接字不是线程安全的),所以我们需要一些类似的东西......
public class SocketTask implements Runnable {
Socket socket;
public SocketTask(){
//create + connect socket here
}
public void run(){
//use socket here
}
}
在应用程序启动时,我们希望初始化工作线程,并希望以某种方式初始化套接字连接......
MyWorkerPool pool = new MyWorkerPool();
for( int i = 0; i < 100; i++)
pool.addWorker( new WorkerThread());
当应用程序请求工作时,我们将任务发送到工作池以立即执行...
pool.queueWork( new SocketTask(..));
使用工作代码更新
基于来自 Gray 和 jontejj 的有用 cmets,我得到了以下代码工作......
SocketTask
public class SocketTask implements Runnable {
private String workDetails;
private static final ThreadLocal<Socket> threadLocal =
new ThreadLocal<Socket>(){
@Override
protected Socket initialValue(){
return new Socket();
}
};
public SocketTask(String details){
this.workDetails = details;
}
public void run(){
Socket s = getSocket(); //gets from threadlocal
//send data on socket based on workDetails, etc.
}
public static Socket getSocket(){
return threadLocal.get();
}
}
ExecutorService
ExecutorService threadPool =
Executors.newFixedThreadPool(5, Executors.defaultThreadFactory());
int tasks = 15;
for( int i = 1; i <= tasks; i++){
threadPool.execute(new SocketTask("foobar-" + i));
}
我喜欢这种方法有几个原因...
- 套接字是可用于正在运行的任务的本地对象(通过 ThreadLocal),从而消除了并发问题。
- 套接字创建一次并保持打开,重复使用 当新任务排队时,消除了套接字对象创建/销毁的开销。
【问题讨论】:
-
感谢工作代码 sn-p,对我帮助很大!我想知道:当线程池关闭时如何关闭数据库连接?
-
搜索orderly shutdown的执行线程,希望对你有帮助。
-
我检查过了,谢谢!不过,我并没有完全帮助我……我正在寻找一种方法来为存储在 ThreadLocal 中的所有数据库连接调用 dbconnection.close()。我真的不知道如何处理这个......我的意思是,一旦线程池被关闭,数据库连接不会自动关闭,对吧?
-
我想我找到了另一个解决方案:我使用一个线程池和一个相同大小的单独向量来存储数据库连接。实际上,在我的情况下,ThreadLocal 并不是真正必要的,因为我并不关心哪个线程获得哪个连接,只要它有一个。无论如何感谢您的灵感!
标签: java concurrency threadpool thread-local threadpoolexecutor