Java Socket编程需关注连接生命周期、异常处理与资源释放,Socket用于客户端通信,ServerSocket仅监听并返回新Socket;须设超时、按序关闭、指定编码及规范换行符。

Java网络编程中的Socket编程基础概念

Java 中的 Socket 编程不是“学完就能直接上生产”的技能,它暴露的是底层 TCP/IP 交互逻辑,一旦忽略连接生命周期、异常分支或资源释放细节,就会出现连接泄漏、SocketException: Broken pipe、线程阻塞甚至 OutOfMemoryError

什么是 Socket,它和 ServerSocket 的分工是什么

Socket 是客户端发起连接并收发数据的端点;ServerSocket 是服务端监听端口、接受连接请求的“守门人”。二者不等价,不能混用:ServerSocket 不发送数据,只调用 accept() 返回一个新的 Socket 实例来代表具体连接。

常见误区:

正确做法是:每个新连接由 accept() 返回独立 Socket,交由单独线程(或 ExecutorService)处理。

Socket 连接建立时最容易卡住的三个地方

默认情况下,Socketconnect()read()write() 全是阻塞操作。没设超时,程序就可能无限等待。

如何安全关闭一个 Socket 连接

关闭顺序和判空比想象中重要。直接调 socket.close() 看似简单,但若输入/输出流已被其他线程提前关闭,再调一次会抛 IOException;若未关闭流,可能触发资源泄漏警告(尤其在 try-with-resources 外手动管理时)。

推荐写法(兼顾健壮性与可读性):

if (socket != null && !socket.isClosed()) {
    try {
        socket.shutdownInput(); // 告诉对方“我不再读”
        socket.shutdownOutput(); // 告诉对方“我不再写”
    } catch (IOException ignored) {}
    try {
        socket.close();
    } catch (IOException ignored) {}
}

关键点:

Socket 编程中常被忽略的字符编码与换行问题

网络传输只认字节,Stringbyte[] 的转换必须指定编码(如 StandardCharsets.UTF_8),否则在中文 Windows 默认 GBK、Linux 默认 UTF-8 下必然乱码。

更隐蔽的是换行符:BufferedReader.readLine() 只识别 \n\r\n\r,但很多协议(如 HTTP、SMTP)严格要求 \r\n。用 println() 发送可能在某些平台变成 \n,导致对端解析失败。

建议统一使用:

真实项目里,90% 的“连得上但收不到数据”问题,根源都在这一行换行符没对齐。

本文转载于:互联网 如有侵犯,请联系zhengruancom@outlook.com删除。
免责声明:正软商城发布此文仅为传递信息,不代表正软商城认同其观点或证实其描述。