客戶端與伺服器的資料傳輸
阿新 • • 發佈:2019-01-24
Java最初是作為網路程式語言出現的,它對網路的高度支援,使得客戶端和伺服器流暢的溝通變成現實。而在網路程式設計中,使用最多的就是Socket,每一個實用的網路程式都少不了它的參與。但Socket的底層機制卻相當棘手,幸運的是,Java平臺給我們一些雖然簡單但是相當強大的類,可以使你更簡單有效的使用Socket開發而無需瞭解底層機制。
請思考一下網路程式設計要解決的問題:而下面這些問題都可以用Socket來解決
- 如何建立兩個節點(電腦)之間的網路連線?
- 如何向另外一個節點(電腦)傳送資訊?
- 如何從外部節點(電腦)接收一個請求並給予響應?
- 如何利用網路協議(TCP、UPD)?
已被廣泛移植到了其他平臺。
Socket實質上提供了程序通訊的端點,網路上的兩個程式通過一個雙向的通訊鏈路實現資料的交換,這個雙向鏈路的一端稱為一個Socket。
ServerSocket類常用方法如下:
O了,講了這麼多理論原理的東西,我來講講編碼方面的知識吧,上述理論只是過一個大概,並不詳細,若對理論感興趣的,可以上網上查詢相關資料:
我寫了一個簡單的例子:一個Socket伺服器,一個是呼叫此伺服器埠的客戶端。以下例子就是客戶端向伺服器的請求。程式碼如下:
- package com.qhs.sockettest;
- import java.io.BufferedReader;
- import java.io.IOException;
- import java.io.InputStream;
- import java.io.InputStreamReader;
- import java.net.ServerSocket;
- import java.net.Socket;
- /**
- * Socket伺服器相關程式碼
- * 編碼完成後,要先執行伺服器開啟埠,然後再執行客戶端。
- */
- public class LoginServer {
- public static void main(String args []){
- try {
- //1.建立一個伺服器Socket繫結一個埠並開始監聽
- ServerSocket serverSocket = new ServerSocket(8800
- //2.使用accept()方法阻塞等待監聽,獲得新的連線
- Socket socket = serverSocket.accept();
- //3.獲得輸入流
- InputStream is = socket.getInputStream();
- BufferedReader br = new BufferedReader(new InputStreamReader(is));
- //4.獲得使用者輸入資訊
- String info = null;
- while(!((info=br.readLine())==null)){
- System.out.println("我是伺服器,使用者資訊為:" + info);
- }
- //5.關閉資源
- br.close();
- is.close();
- socket.close();
- serverSocket.close();
- } catch (IOException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- }
- }
- package com.qhs.sockettest;
- import java.io.IOException;
- import java.io.OutputStream;
- import java.io.PrintWriter;
- import java.net.Socket;
- import java.net.UnknownHostException;
- /**
- * 客戶端
- */
- public class LoginClient {
- public static void main(String args []){
- try {
- //1.建立客服端Socket連線,指定伺服器的位置以及埠
- Socket socket = new Socket("localhost", 8800);
- //2.得到Socket讀寫流
- OutputStream os = socket.getOutputStream();
- PrintWriter pw = new PrintWriter(os);
- //3.利用流按照一定協議對Socket進行讀/寫操作
- String info = "使用者名稱:Jeff;使用者密碼:123456";
- pw.write(info);
- pw.flush();
- //4.關閉資源
- pw.close();
- os.close();
- socket.close();
- } catch (UnknownHostException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- } catch (IOException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- }
- }
測試進行時,請先執行伺服器的程式碼,使得埠開啟。然後再執行客戶端去連線。最後在控制檯輸入的結果如圖:
上面的例子就比如一個人打電話,另外一個人只是聽,不說話,單向通訊。所以執行後,後臺只有客戶端有伺服器的資訊
接下來,我們在上述程式碼中新增一些程式碼,使得伺服器也有響應,等於雙方相互通訊的目的:
伺服器程式碼:
- package com.qhs.sockettest;
- 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;
- /**
- * Socket伺服器相關程式碼
- * 編碼完成後,要先執行伺服器開啟埠,然後再執行客戶端。
- */
- public class LoginServer {
- public static void main(String args []){
- try {
- //1.建立一個伺服器Socket繫結一個埠並開始監聽
- ServerSocket serverSocket = new ServerSocket(8800);
- //2.使用accept()方法阻塞等待監聽,獲得新的連線
- Socket socket = serverSocket.accept();
- //3.獲得輸入流
- InputStream is = socket.getInputStream();
- BufferedReader br = new BufferedReader(new InputStreamReader(is));
- //伺服器的響應加進來的程式碼:獲得輸出流
- OutputStream os = socket.getOutputStream();
- PrintWriter pw = new PrintWriter(os);
- //4.獲得使用者輸入資訊
- String info = null;
- while(!((info=br.readLine())==null)){
- System.out.println("我是伺服器,使用者資訊為:" + info);
- }
- //新增程式碼:給客戶一個響應
- String reply = "服務的響應>>>>Welcome!";
- pw.write(reply);
- pw.flush();
- //5.關閉資源
- pw.close();
- os.close();
- br.close();
- is.close();
- socket.close();
- serverSocket.close();
- } catch (IOException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- }
- }
客戶端程式碼:
- package com.qhs.sockettest;
- 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.Socket;
- import java.net.UnknownHostException;
- /**
- * 客戶端
- */
- public class LoginClient {
- public static void main(String args []){
- try {
- //1.建立客服端Socket連線,指定伺服器的位置以及埠
- Socket socket = new Socket("localhost", 8800);
- //2.得到Socket讀寫流
- OutputStream os = socket.getOutputStream();
- PrintWriter pw = new PrintWriter(os);
- //新增程式碼:輸入流
- InputStream is = socket.getInputStream();
- BufferedReader br = new BufferedReader(new InputStreamReader(is));
- //3.利用流按照一定協議對Socket進行讀/寫操作
- String info = "使用者名稱:Jeff;使用者密碼:123456";
- pw.write(info);
- pw.flush();
- socket.shutdownOutput(); //關閉socket
- //新增程式碼:接收伺服器響應並列印顯示
- String reply = null;
- while(!((reply=br.readLine())==null)){
- System.out.println("我是客戶端,接收到伺服器的響應為:" + reply);
- }
- //4.關閉資源
- br.close();
- is.close();
- pw.close();
- os.close();
- socket.close();
- } catch (UnknownHostException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- } catch (IOException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- }
- }
我們執行測試一下,先執行伺服器,再執行客戶端。我們會看到,這次後臺不只是客戶端接收到了伺服器給的請求資訊,在伺服器後臺,也輸出了相應的響應資訊:我們可以點選下圖所示的切換來檢視客戶端與伺服器接收與響應的後臺資訊:
最後,我們來總結一下Socket中伺服器與客戶端的各步驟。
伺服器端:
- 建立一個伺服器Socket繫結指定的埠並開始監聽;
- 使用accept()方法阻塞等待監聽,獲取新的連線;
- 建立輸入和輸出流;
- 在已有的協議上產生會話;
- 使用close()方法關閉流和Socket;
- 建立客戶端Socket連線,指定伺服器的位置以及埠;
- 得到Socket的讀寫流;
- 利用流按照一定的協議對Socket進行讀/寫操作;
- 使得close()方法關閉流和Socket
這是Socket網路程式設計之傳遞字元型別,接下來我會在我部落格另外一篇中Socket升級傳遞物件程式設計