Java线程的传说(3)——如何关闭一个正在accept的ServerSocket?

本文介绍了如何使用Java并发处理网络服务,通过创建线程池和实现Runnable接口来处理网络请求,并提供了关闭线程池的方法以优雅地结束服务。重点讨论了如何避免资源泄露和错误处理,确保服务稳定运行。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

加入一个ServerSocket正在另一个线程堵塞accept,那如何停止accept或者关闭Socket?Server socket 设置下超时 setSoTimeout 然后在Listen线程中用interrupt其实直接close socket也可以,不过会抛出异常,我的意思是有什么比较安全而又简单的办法?难道要加一个标志,然后要关闭的时候把标志设为stop,然后连接listernSocket?这样也未免太麻烦了JDK5.0里面新增了java.util.concurrent包(对于多线程的开发建议尽量使用这个包),下面是javadoc里面的样例代码用法示例下面给出了一个网络服务的简单结构,这里线程池中的线程作为传入的请求。它使用了预先配置的Executors.newFixedThreadPool(int) 工厂方法:
 
java 代码
 
class NetworkService implements Runnable { 
   private final ServerSocket serverSocket; 
   private final ExecutorService pool; 
 
   public NetworkService(int port, int poolSize) 
       throws IOException { 
     serverSocket = new ServerSocket(port); 
     pool = Executors.newFixedThreadPool(poolSize); 
   } 
 
   public void run() { // run the service 
     try { 
       for (;;) { 
         pool.execute(new Handler(serverSocket.accept())); 
       } 
     } catch (IOException ex) { 
       pool.shutdown(); 
     } 
   } 
 } 
 
 class Handler implements Runnable { 
   private final Socket socket; 
   Handler(Socket socket) { this.socket = socket; } 
   public void run() { 
     // read and service request on socket 
   } 
}

 
下列方法分两个阶段关闭 ExecutorService。第一阶段调用 shutdown 拒绝传入任务,然后调用 shutdownNow(如有必要)取消所有遗留的任务:

java 代码
 
void shutdownAndAwaitTermination(ExecutorService pool) { 
  pool.shutdown(); // Disable new tasks from being submitted 
  try { 
    // Wait a while for existing tasks to terminate 
    if (!pool.awaitTermination(60, TimeUnit.SECONDS)) { 
      pool.shutdownNow(); // Cancel currently executing tasks 
      // Wait a while for tasks to respond to being cancelled 
      if (!pool.awaitTermination(60, TimeUnit.SECONDS)) 
          System.err.println("Pool did not terminate"); 
    } 
  } catch (InterruptedException ie) { 
    // (Re-)Cancel if current thread also interrupted 
    pool.shutdownNow(); 
    // Preserve interrupt status 
    Thread.currentThread().interrupt(); 
  } 

内存一致性效果:线程中向 ExecutorService 提交 Runnable 或 Callable 任务之前的操作 happen-before 由该任务所提取的所有操作,后者依次 happen-before 通过 Future.get() 获取的结果。

软件包 java.util.concurrent

http://www.jar114.com/jdk6/zh_CN/api/java/util/concurrent/package-summary.html

要实现多个客户端同时连接服务器进行通信,可以使用Java线程线程池的概念。以下是一个示例代码: 服务器端代码: ```java import java.io.*; import java.net.*; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; public class Server { public static void main(String[] args) throws IOException { ServerSocket serverSocket = new ServerSocket(12345); ExecutorService executorService = Executors.newFixedThreadPool(10); while (true) { System.out.println("服务器等待建立连接..."); Socket socket = serverSocket.accept(); // 每次有客户端建立连接就创建一个新的线程处理 executorService.execute(new ClientHandler(socket)); System.out.println("连接成功!"); } } } class ClientHandler implements Runnable { private Socket socket; public ClientHandler(Socket socket) { this.socket = socket; } @Override public void run() { try { BufferedReader br = new BufferedReader(new InputStreamReader(socket.getInputStream())); BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(socket.getOutputStream())); String line; while ((line = br.readLine()) != null) { System.out.println("接收到客户端消息:" + line); // 处理客户端发送的消息 bw.write("服务器收到消息:" + line); bw.newLine(); bw.flush(); } br.close(); bw.close(); socket.close(); } catch (IOException e) { e.printStackTrace(); } } } ``` 客户端代码: ```java import java.io.*; import java.net.*; public class Client { public static void main(String[] args) throws IOException { Socket socket = new Socket("服务器IP地址", 12345); BufferedReader br = new BufferedReader(new InputStreamReader(socket.getInputStream())); BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(socket.getOutputStream())); // 发送消息给服务器 bw.write("Hello, Server!"); bw.newLine(); bw.flush(); // 接收服务器的响应 String response = br.readLine(); System.out.println("服务器响应:" + response); br.close(); bw.close(); socket.close(); } } ``` 在服务器端代码中,通过创建一个线程池(这里使用固定大小的线程池),每当有一个客户端连接时,就创建一个线程去处理客户端的请求。该线程负责与客户端进行通信。 在客户端代码中,创建一个Socket对象与服务器建立连接,然后通过输入输出流进行通信。在示例中,客户端向服务器发送一条消息,服务器接收到后处理,然后返回响应给客户端。 这样,多个客户端可以同时连接服务器进行通信,服务器通过多线程来处理每个客户端的请求,并保证线程安全性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值