Java Socket編程(二)
發表時間:2023-08-13 來源:明輝站整理相關軟件相關文章人氣:4
[摘要]Java面向連接的類Sockets有兩種主要的操作方式:面向連接的和無連接的.面向連接的sockets操作就像一部電話,他們必須建立一個連接和一人呼叫.所有的事情在到達時的順序與它們出發時的順序時一...
Java面向連接的類
Sockets有兩種主要的操作方式:面向連接的和無連接的.面向連接的sockets操作就像一部電話,他們必須建立一個連接和一人呼叫.所有的事情在到達時的順序與它們出發時的順序時一樣.無連接的sockets操作就像是一個郵件投遞,,沒有什么保證,多個郵件可能在到達時的順序與出發時的順序不一樣.
到底用哪種模式是郵應用程序的需要決定的.如果可靠性更重要的話,用面向連接的操作會好一些.比如文件服務器需要他們的數據的正確性和有序性.如果一些數據丟失了,系統的有效性將會失去.一些服務器,比如間歇性地發送一些數據塊.如果數據丟了的話,服務器并不想要再重新發過一次.因為當數據到達的時候,它可能已經過時了.確保數據的有序性和正確性需要額外的操作的內存消耗,額外的費用將會降低系統的回應速率.
無連接的操作使用數據報協議.一個數據報是一個獨立的單元,它包含了所有的這次投遞的信息.把它想象成一個信封吧,它有目的地址和要發送的內容.這個模式下的socket不需要連接一個目的的socket,它只是簡單地投出數據報.無連接的操作是快速的和高效的,但是數據安全性不佳.
面向連接的操作使用TCP協議.一個這個模式下的socket必須在發送數據之前與目的地的socket取得一個連接.一旦連接建立了,sockets就可以使用一個流接口:打開-讀-寫-關閉.所有的發送的信息都會在另一端以同樣的順序被接收.面向連接的操作比無連接的操作效率更低,但是數據的安全性更高.
SUN一直是網絡建設的支持者,所以在Java中支持sockets就不足為奇了.實際上,Java降低了建立一個sockets程序的難度.每一個傳輸模式都被封裝到了不同的類中.面向連接的類將會首先被我們討論.
在Java中面向連接的類有兩種形式,它們分別是客戶端和服務器端.客戶端這一部分是最簡單的,所以我們先討論它.
列表9.1列出了一個簡單的客戶端的程序.它向一個服務器發出一個請求,取回一個HTML文檔,并把它顯示在控制臺上.
9.1一個簡單的socket客戶端
import java.io.*;
import java.net.*;
/**
* 一個簡單的從服務器取回一個HTML頁面的程序
* 注意:merlin是本地機器的名字
*/
public class SimpleWebClient {
public static void main(String args[])
{
try
{
// 打開一個客戶端socket連接
Socket clientSocket1 = new Socket("merlin", 80);
System.out.println("Client1: " + clientSocket1);
// 取得一個網頁
getPage(clientSocket1);
}
catch (UnknownHostException uhe)
{
System.out.println("UnknownHostException: " + uhe);
}
catch (IOException ioe)
{
System.err.println("IOException: " + ioe);
}
}
/**
*通過建立的連接請求一個頁面,顯示回應然后關閉socket
*/
public static void getPage(Socket clientSocket)
{
try
{
// 需要輸入和輸出流
DataOutputStream outbound = new DataOutputStream(
clientSocket.getOutputStream() );
DataInputStream inbound = new DataInputStream(
clientSocket.getInputStream() );
// 向服務器發出HTTP請求
outbound.writeBytes("GET / HTTP/1.0\r\n\r\n");
// 讀出回應
String responseLine;
while ((responseLine = inbound.readLine()) != null)
{
// 把每一行顯示出來
System.out.println(responseLine);
if ( responseLine.indexOf("") != -1 )
break;
}
// 清除
outbound.close();
inbound.close();
clientSocket.close();
}
catch (IOException ioe)
{
System.out.println("IOException: " + ioe);
}
}
}
回憶一個,一個客戶端向一個正在監聽的服務器socket發出一個連接.客戶端的sockets是用Socket類建立的.下面的程序建立了一個客戶端的socket并且連接到了一個主機:
Socket clientSocket = new Socket("merlin", 80);
第一個參數是你想要連接的主機的名稱,第二個參數是端口號.一個主機名稱指定了目的的名稱.端口號指定了由哪個應用程序來接收.在我們的情況下,必須指定80,因為它是默認的HTTP協議的端口.另外的知名的端口列在表9.1中,看:
知名的端品:
echo 7
daytime 13
daytime 13
ftp 21
telnet 23
smtp 25
finger 79
http 80
pop3 110
因為Socket類是面向連接的,它提供了一個可供讀寫的流接口.java.io包中的類可以用來訪問一個已連接的socket:
DataOutputStream outbound = new DataOutputStream(
clientSocket.getOutputStream() );
DataInputStream inbound = new DataInputStream( clientSocket.getInputStream()
);
一旦流建立了,一般的流操作就可以做了:
outbound.writeBytes("GET / HTTP/1.0\r\n\r\n);
String responseLine;
while ( (responseLine = inbound.readLine()) != null)
{
System.out.println(responseLine);
}
以上的小程序請求了一個WEB頁面并且把它顯示出來.當程序完成之后,連接必須關閉.
outbound.close();
inbound.close();
clientSocket.close();
注意socket流必須首先關閉.所有的的socket流必須在socket關閉之前關閉.這個小程序非常地簡單,但是所有的客戶端程序都必須遵首下面的基本的步驟:
1.建立客戶端socket連接.
2.得到socket的讀和寫的流.
3.利用流.
4.關閉流.
5.關閉socket.
使用一個服務器端的socket只是有一點復雜,它將在下面講到.