Java Socket通信介紹及可能遇到的問(wèn)題解決
前言
本文主要給大家介紹了關(guān)于Java中Socket通信的相關(guān)內(nèi)容,分享出來(lái)供大家參考學(xué)習(xí),下面話(huà)不多說(shuō)了,來(lái)一起看看詳細(xì)的介紹吧。
Java中基于TCP協(xié)議實(shí)現(xiàn)網(wǎng)絡(luò)通信的兩個(gè)類(lèi):客戶(hù)端的Socket和服務(wù)器端的ServerSocket。
Socket通信模型如圖所示:

不管Socket通信的功能有多復(fù)雜,任何socket通信過(guò)程的基本結(jié)構(gòu)都是一樣的。
其基本步驟為:
①分別在客戶(hù)端和服務(wù)器端創(chuàng)建Socket和ServerSocket實(shí)例;服務(wù)器端通過(guò).accept()方法等待請(qǐng)求并阻塞。請(qǐng)求收到后,建立連接Socket對(duì)象。
②通過(guò)getInputStream和getOutputStream方法分別在客戶(hù)端和服務(wù)器端打開(kāi)輸入輸出流
③利用IO流進(jìn)行讀寫(xiě)操作
④關(guān)閉所有的流資源和套接字資源。
其中,編程工作主要集中在第三步,其他的部分代碼基本相同。所有步驟都可能拋出IO異常!
我在編寫(xiě)一個(gè)簡(jiǎn)單的socket程序時(shí),使用的Socket通信出現(xiàn)了一個(gè)問(wèn)題:我在客戶(hù)端寫(xiě)入的數(shù)據(jù),在服務(wù)器端無(wú)法輸出。當(dāng)我從客戶(hù)端斷開(kāi)連接時(shí),之前寫(xiě)入的所有數(shù)據(jù)立刻在服務(wù)器端輸出出來(lái)了。經(jīng)過(guò)反復(fù)的驗(yàn)證和求解,以下是我的結(jié)論和解決方法。希望有同樣問(wèn)題的小伙伴看完可以解決問(wèn)題。
通過(guò)一端的Socket建立了PrintWriter類(lèi)來(lái)寫(xiě)入數(shù)據(jù),通過(guò)另一端的Socket建立了BufferedReader類(lèi)來(lái)讀取數(shù)據(jù)并輸出。
如果數(shù)據(jù)寫(xiě)入后沒(méi)有被顯示,可能的原因有兩種:
一、寫(xiě)入的數(shù)據(jù)存儲(chǔ)在緩沖區(qū)中,沒(méi)有被寫(xiě)入IO流中:
如果不主動(dòng)的干涉,寫(xiě)入的數(shù)據(jù)會(huì)一直堆在緩沖區(qū)中,直到緩沖區(qū)滿(mǎn)了引發(fā)JVM自動(dòng)刷新緩沖區(qū)。顯然這不符合我們的需求。對(duì)于這種情況,PrintWriter類(lèi)提供了flush()方法來(lái)強(qiáng)制刷新緩沖區(qū),將緩沖區(qū)數(shù)據(jù)寫(xiě)入IO流中。另外,PrintWriter類(lèi)的構(gòu)造器有一個(gè)參數(shù)”boolean autoflush“,這個(gè)參數(shù)默認(rèn)為false,如果設(shè)置為true,則會(huì)開(kāi)啟自動(dòng)刷新緩沖區(qū)功能。但是請(qǐng)注意,這里的自動(dòng)刷新是有觸發(fā)條件的,那就是:PrintWriter類(lèi)寫(xiě)入數(shù)據(jù)的方法必須是println、printf或者format方法時(shí),才會(huì)觸發(fā)自動(dòng)刷新。如果是調(diào)用write()這類(lèi)方法寫(xiě)入數(shù)據(jù),是不會(huì)觸發(fā)自動(dòng)刷新的!總結(jié)起來(lái),就是三點(diǎn):autoflush參數(shù)設(shè)置,write和println方法的選擇,flush方法的使用。對(duì)這三個(gè)進(jìn)行組合,就能保證在Socket通信的某一端寫(xiě)入數(shù)據(jù)時(shí),數(shù)據(jù)一定能成功地寫(xiě)入到IO流中!
二、讀取數(shù)據(jù)使用了readLine()方法,該方法沒(méi)有正常的結(jié)束:
請(qǐng)注意,BufferedReader類(lèi)的readLine()方法是一個(gè)阻塞函數(shù)!也就是說(shuō),這個(gè)方法本身是讀取一行數(shù)據(jù),但是它自己識(shí)別不了什么叫做“一行”!當(dāng)調(diào)用該方法讀取完一段數(shù)據(jù)后,它會(huì)阻塞,而不會(huì)return它的讀取數(shù)據(jù)。這就是為什么有的時(shí)候明明已經(jīng)刷新了緩沖區(qū)正確的寫(xiě)入數(shù)據(jù)了,還是通過(guò)輸入流讀取數(shù)據(jù)并顯示出來(lái)的原因。
對(duì)于readLine()方法,它解除阻塞、正確結(jié)束并返回讀取的值,只有以下幾種情況:
①讀取的數(shù)據(jù)里含有回車(chē)符"\r"或者換行符"\n"或者回車(chē)換行符"\r\n";
②讀取的數(shù)據(jù)是在另一端通過(guò)println方法寫(xiě)入的,因?yàn)閜rintln方法自帶換行符;
③BufferedReader類(lèi)的緩沖區(qū)滿(mǎn)了,那么JVM會(huì)自動(dòng)刷新緩沖區(qū)從而釋放“積攢”的數(shù)據(jù)(但是鑒于默認(rèn)緩沖區(qū)大小為8192個(gè)字符,對(duì)于小數(shù)據(jù)量的通信,顯然觸發(fā)不了);
④對(duì)于讀取的數(shù)據(jù),寫(xiě)入這些數(shù)據(jù)的流發(fā)生異?;蛘咧苯雨P(guān)閉,那么readLine()就會(huì)把它吃的數(shù)據(jù)全部吐出來(lái)。這就剛好解釋了,為什么在我的程序中,斷開(kāi)客戶(hù)端Socket連接,服務(wù)器端立刻輸出所有客戶(hù)端消息的原因。
綜上,在Socket通信過(guò)程中,保證某一端輸出流的緩沖被刷新,保證另一端的readLine方法能正常停止,即可解決寫(xiě)入的數(shù)據(jù)在另一端無(wú)法輸出的問(wèn)題。
以下是我修改后能成功運(yùn)行的代碼,分別是服務(wù)器端Socket和客戶(hù)端Socket。
over!
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.PrintWriter;
import java.net.ServerSocket;
import java.net.Socket;
import java.io.BufferedWriter;
import java.io.OutputStreamWriter;
public class ShakingServer{
public static void main(String[] args) throws IOException {
//創(chuàng)建服務(wù)器套接字實(shí)例,設(shè)置監(jiān)聽(tīng)端口為2000
ServerSocket server=new ServerSocket(2000);
//開(kāi)始監(jiān)聽(tīng)客戶(hù)端的請(qǐng)求,并阻塞
Socket socket=server.accept();
//請(qǐng)求收到后,自動(dòng)建立連接。通過(guò)IO流進(jìn)行數(shù)據(jù)傳輸
System.out.println("連接建立成功");
OutputStream os=socket.getOutputStream();
PrintWriter pw=new PrintWriter(new BufferedWriter(new OutputStreamWriter(os)),true);
pw.write("歡迎訪問(wèn)搖頭耶穌的世界!");
pw.flush();
//因?yàn)槲谊P(guān)閉了輸出流,所以另一端的readLine方法才正常結(jié)束了
socket.shutdownOutput();
InputStream is=socket.getInputStream();
InputStreamReader isr=new InputStreamReader(is);
BufferedReader br=new BufferedReader(isr);
while(true) {
String str=br.readLine();
if(str.equals("quit")) {
break;
}
System.out.println("Client said: "+str);
}
socket.shutdownInput();
//socket.shutdownOutput();
socket.close();
server.close();
}
}
import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.OutputStreamWriter;
import java.io.PrintWriter;
import java.net.Socket;
public class ShakingClient{
public static void main(String[] args) throws IOException{
//創(chuàng)建客戶(hù)端的套接字,設(shè)置連接的服務(wù)器的IP地址和端口號(hào)
Socket socket=new Socket("169.254.132.203",2000);
//輸入流讀取服務(wù)器發(fā)送的信息
BufferedReader br=new BufferedReader(new InputStreamReader(socket.getInputStream()));
//開(kāi)啟自動(dòng)刷新緩沖區(qū)
PrintWriter pw=new PrintWriter(new BufferedWriter(new OutputStreamWriter(socket.getOutputStream())),true);
//從鍵盤(pán)讀取數(shù)據(jù)
BufferedReader ii=new BufferedReader(new InputStreamReader(System.in));
System.out.println(br.readLine());
//因?yàn)殚_(kāi)啟了自動(dòng)刷新,且調(diào)用的是println方法,所以可以不調(diào)用flush方法
pw.println("請(qǐng)求進(jìn)入搖頭耶穌的世界");
//pw.flush();
while(true) {
String str=ii.readLine();
//使用了回車(chē)符來(lái)保證另一端的readLine方法正常結(jié)束
pw.write(str+"\r");
pw.flush();
//如果輸入quit則退出聊天室
if(str.equals("quit")) {
break;
}
}
socket.shutdownInput();
socket.shutdownOutput();
socket.close();
}
}
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
相關(guān)文章
java+opencv實(shí)現(xiàn)人臉識(shí)別功能
這篇文章主要介紹了java+opencv實(shí)現(xiàn)人臉識(shí)別功能,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-05-05
Java工程使用ffmpeg進(jìn)行音視頻格式轉(zhuǎn)換的實(shí)現(xiàn)
FFmpeg是一套可以用來(lái)記錄、轉(zhuǎn)換數(shù)字音頻、視頻,并能將其轉(zhuǎn)化為流的開(kāi)源計(jì)算機(jī)程序,本文主要介紹了Java工程使用ffmpeg進(jìn)行音視頻格式轉(zhuǎn)換的實(shí)現(xiàn)2024-02-02
Java如何通過(guò)"枚舉的枚舉"表示二級(jí)分類(lèi)的業(yè)務(wù)場(chǎng)景
這篇文章主要介紹了Java如何通過(guò)"枚舉的枚舉"表示二級(jí)分類(lèi)的業(yè)務(wù)場(chǎng)景問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-06-06
WebClient拋UnsupportedMediaTypeException異常解決
這篇文章主要為大家介紹了WebClient拋UnsupportedMediaTypeException異常的解決方案,文中給大家介紹了六中方案,有需要的朋友可以借鑒參考下,希望能夠有所幫助2022-02-02
Java 中的 @SneakyThrows 注解的使用方法(簡(jiǎn)化異常處理的利與弊)
@SneakyThrows是Lombok提供的一個(gè)注解,用于簡(jiǎn)化Java方法中的異常處理,特別是對(duì)于檢查型異常,它允許方法拋出異常而不必顯式聲明或捕獲這些異常,本文介紹Java 中的 @SneakyThrows 注解的使用方法,感興趣的朋友一起看看吧2025-03-03
java簡(jiǎn)單實(shí)現(xiàn)自定義日歷
這篇文章主要為大家詳細(xì)介紹了java簡(jiǎn)單實(shí)現(xiàn)自定義日歷,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-05-05
基于@PostConstruct注解的使用,解決向靜態(tài)變量注入值
這篇文章主要介紹了基于@PostConstruct注解的使用,解決向靜態(tài)變量注入值問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09

