淺談java socket的正確關(guān)閉姿勢(shì)
java socket對(duì)應(yīng)的是網(wǎng)絡(luò)協(xié)議中的tcp,tcp的三次握手、四次揮手、11中狀態(tài)什么的這里就不說(shuō)了,不知道大家平常使用socket的時(shí)候如果不注意的情況下,會(huì)不會(huì)遇到各種異常報(bào)錯(cuò)。
例如:
java.net.SocketException:socket is closed
錯(cuò)誤提示的出現(xiàn)場(chǎng)景:
自己主動(dòng)關(guān)閉了socket,但是之后還從里面讀寫(xiě)數(shù)據(jù)
Software caused connection abort: socket write error
錯(cuò)誤提示的出現(xiàn)場(chǎng)景:
對(duì)方已經(jīng)關(guān)閉socket,依舊向?qū)Ψ綄?xiě)數(shù)據(jù)
connection reset (by peer)
錯(cuò)誤提示出現(xiàn)的場(chǎng)景:
一端socket被關(guān)閉,另一端仍然發(fā)送數(shù)據(jù),發(fā)送的第一個(gè)數(shù)據(jù)包 connection reset by peer
一端socket退出,退出時(shí)為關(guān)閉連接,另一端讀數(shù)據(jù) connection reset
所以在使用socket時(shí),需要約定好雙方讀寫(xiě)完成的條件,然后關(guān)閉輸入輸出流:
socket.shutdownInput(); socket.shutdownOutput();
即當(dāng)一方寫(xiě)入完成后,調(diào)用shutdownOutput關(guān)閉輸出流,這時(shí)候?qū)Ψ降膔ead方法就會(huì)返回-1,這時(shí)候?qū)Ψ骄椭滥銓?xiě)完了,對(duì)方可以關(guān)閉輸入流,然后等待對(duì)方寫(xiě)入完成調(diào)用shutdownOutput后己方再調(diào)用shutdownInput,雙方就正常關(guān)閉了輸入輸出流,這時(shí)候socket就不會(huì)出現(xiàn)異常了。
下面是一個(gè)socket交互的例子:
server端
public class OioServer {
public static void main(String[] args) throws IOException {
ServerSocket serverSocket = new ServerSocket(8080);
while (true) {
Socket socket = serverSocket.accept();
System.out.println("socket = " + socket);
new Thread(() -> {
try {
InputStream in = socket.getInputStream();
OutputStream out = socket.getOutputStream();
out.write("hello! I get your message that is follow".getBytes(Charset.forName("UTF-8")));
byte[] buf = new byte[1024];
int len;
while ((len = in.read(buf)) != -1) {
System.out.print(new String(buf, 0, len, Charset.forName("UTF-8")));
out.write(buf, 0, len);
}
out.write("\n end \n".getBytes(Charset.forName("UTF-8")));
out.flush();
socket.shutdownInput();
socket.shutdownOutput();
} catch (IOException e) {
e.printStackTrace();
}finally {
try {
socket.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}).start();
}
}
}
client端
public class OioClient {
public static void main(String[] args) throws IOException {
Socket socket = new Socket("127.0.0.1", 8080);
InputStream in = socket.getInputStream();
new Thread(() -> {
BufferedInputStream bufferIn = new BufferedInputStream(in);
byte[] buf = new byte[1024];
try {
int len;
while ((len = bufferIn.read(buf)) != -1) {
System.out.print(new String(buf, 0, len, Charset.forName("UTF-8")));
}
}catch (Exception e) {
e.printStackTrace();
}
try {
socket.shutdownInput();
socket.close();
} catch (IOException e) {
e.printStackTrace();
}
}).start();
OutputStream out = socket.getOutputStream();
int cout = 10;
while (cout-- > 0) {
out.write(("this time is " + System.currentTimeMillis() + "\n").getBytes("UTF-8"));
}
socket.shutdownOutput();
}
}
java socket - 半關(guān)閉
通常,使用關(guān)閉輸出流來(lái)表示輸出已經(jīng)結(jié)束。但在進(jìn)行網(wǎng)絡(luò)通信時(shí)則不能這樣做。因?yàn)槲覀冴P(guān)閉輸出流時(shí),該輸出流對(duì)應(yīng)的Socket也將隨之關(guān)閉,這樣程序?qū)o(wú)法再?gòu)脑搒ocket中讀取數(shù)據(jù)。
為了應(yīng)付這種情況,socket提供了兩個(gè)半關(guān)閉的方法用來(lái)只關(guān)閉socket的輸入流或者輸出流,用以表示輸出數(shù)據(jù)已經(jīng)發(fā)送完成。
方法詳情:
shutdownInput():關(guān)閉該socket的輸入流,程序還可以通過(guò)該socket的輸出流輸出數(shù)據(jù);
shutdownOutput():關(guān)閉該socket的輸出流,程序還可以通過(guò)該socket的輸入流讀取數(shù)據(jù)。
當(dāng)調(diào)用shutdownInput()或shutdownOutput()方法關(guān)閉輸入流或輸出流后,該socket處于半關(guān)閉狀態(tài)。
此時(shí)可以使用isInputShutdown()或isOutputShutdown()來(lái)判斷該socket是否處于半讀狀態(tài)或半寫(xiě)狀態(tài)。
需要注意的是,即使同一個(gè)socket先后調(diào)用shutdownInput()和shutdownInput()方法,該socket實(shí)例仍然沒(méi)有被關(guān)閉,只是該socket既不能輸出數(shù)據(jù)也不能讀取數(shù)據(jù)而已。
當(dāng)調(diào)用shutdownInput()或shutdownOutput()方法關(guān)閉了輸入流或輸出流之后,該socket無(wú)法再次打開(kāi)輸出流或輸入流,因此這種做法不適合需要保持持久通信狀態(tài)的交互式應(yīng)用。
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
SpringBoot yaml語(yǔ)法與數(shù)據(jù)讀取操作詳解
YAML 是 “YAML Ain’t Markup Language”(YAML 不是一種標(biāo)記語(yǔ)言)的遞歸縮寫(xiě)。在開(kāi)發(fā)的這種語(yǔ)言時(shí),YAML 的意思其實(shí)是:“Yet Another Markup Language”(仍是一種標(biāo)記語(yǔ)言),本文給大家介紹的非常詳細(xì),需要的朋友可以參考下2022-07-07
springboot jackson自定義序列化和反序列化實(shí)例
這篇文章主要介紹了spring boot jackson自定義序列化和反序列化實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-10-10
SpringBoot接口返回結(jié)果封裝方法實(shí)例詳解
在實(shí)際項(xiàng)目中,一般會(huì)把結(jié)果放在一個(gè)封裝類中,封裝類中包含http狀態(tài)值,狀態(tài)消息,以及實(shí)際的數(shù)據(jù)。這里主要記錄兩種方式,通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),需要的朋友參考下吧2021-09-09
springboot2.X整合prometheus監(jiān)控的實(shí)例講解
這篇文章主要介紹了springboot2.X整合prometheus監(jiān)控的實(shí)例講解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2021-03-03
使用spring boot開(kāi)發(fā)時(shí)java對(duì)象和Json對(duì)象轉(zhuǎn)換的問(wèn)題
這篇文章主要介紹了使用spring boot開(kāi)發(fā)時(shí)java對(duì)象和Json對(duì)象轉(zhuǎn)換的問(wèn)題,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-03-03
Java實(shí)例講解多態(tài)數(shù)組的使用
本文章向大家介紹Java多態(tài)數(shù)組,主要包括Java多態(tài)數(shù)組使用實(shí)例、基本知識(shí)點(diǎn)總結(jié)和需要注意事項(xiàng),具有一定的參考價(jià)值,需要的朋友可以參考一下2022-05-05
Java中關(guān)于int和Integer的區(qū)別詳解
本篇文章小編為大家介紹,在Java中 關(guān)于int和Integer的區(qū)別詳解,需要的朋友參考下2013-04-04
java實(shí)現(xiàn)學(xué)生宿舍系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)學(xué)生宿舍系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-03-03

