核心是ServerSocket接收连接后交由独立线程处理,用CopyOnWriteArrayList安全广播,显式指定UTF-8编码并及时捕获IOException检测断连。

怎么让多个客户端通过服务器收发消息
核心是用 ServerSocket 接收连接,每个客户端分配一个独立的 Thread 处理读写。服务器不能阻塞在某个客户端上,否则新连接会被卡住。
- 服务器用
serverSocket.accept()获取Socket后,立刻丢给新线程处理,主线程继续等待下一个连接 - 每个客户端线程维护自己的
BufferedReader和PrintWriter,从输入流读消息,往所有其他客户端的输出流写消息 - 别直接在主线程里循环调用
readLine()—— 一旦某个客户端断开或发空行,容易抛NullPointerException或卡死
怎么安全地广播消息给所有在线客户端
不能把所有 PrintWriter 存在普通 List 里然后遍历写——并发写会出错,而且断连的客户端没清理会导致 IOException。
- 用
Collections.synchronizedList(new ArrayList())包一层,或者更稳妥地用CopyOnWriteArrayList - 每次广播前检查
printWriter.checkError(),为true就说明底层 socket 已断,应从列表中移除 - 写之前加
if (pw != null && pw.isClosed() == false)防空指针和已关闭流 - 移除操作必须同步,比如用
synchronized(clients) { clients.remove(pw); }
客户端异常断连时服务器怎么及时发现
TCP 连接断开不会立刻通知服务端,得靠读操作触发异常,或者加心跳检测。
- 客户端线程里对
bufferedReader.readLine()做 try-catch,捕获IOException(如 connection reset、broken pipe)就视为断连 - 不要依赖
socket.isClosed()或isConnected()—— 它们返回的是本地状态,不是真实网络状态 - 如果想更主动,可以让客户端每 30 秒发个
"PING",服务器收到后回"PONG";超时未收到 PING 就踢掉该连接
为什么客户端发中文显示乱码
根本原因是字符集不一致:客户端用 UTF-8 写,服务器用平台默认编码(如 Windows 的 GBK)读,就会解码失败。
立即学习“Java免费学习笔记(深入)”;
- 创建
InputStreamReader和OutputStreamWriter时,显式指定"UTF-8",例如:new InputStreamReader(socket.getInputStream(), "UTF-8") - 客户端也一样,
PrintWriter构造时传true自动 flush,并指定编码:new PrintWriter(new OutputStreamWriter(socket.getOutputStream(), "UTF-8"), true) - IDE 运行配置里也要确认控制台编码是 UTF-8,否则即使逻辑正确,控制台也显示???









