java如何實(shí)現(xiàn)socket連接方法封裝
Java實(shí)現(xiàn)socket連接技巧
Socket通信幾乎無(wú)時(shí)不在,當(dāng)然能夠搜集到的信息也大量存在, 為了避免重復(fù)的勞作, 抽取了關(guān)于客戶端和服務(wù)端的Socket, 并將其應(yīng)用到適合JVM(LInux/Windows)或者DVM(Android)平臺(tái)。
這個(gè)封裝好的API具有以下優(yōu)勢(shì):
1.滿足具有Socket客戶端需求的基本應(yīng)用。
2.滿足具有Socket服務(wù)端的基本應(yīng)用。具備并發(fā)能力, 能滿足可設(shè)定個(gè)數(shù)客戶端連接。
本文的目的就是為了對(duì)Socket做一個(gè)封裝, 方便客戶端和服務(wù)端能直接使用Socket.封裝好的API可以從下面獲取
Java Socket的封裝
其中src/中的是API源碼; usage/目錄是使用例程
1 客戶端Socket API要點(diǎn)
1)客戶端和指定的服務(wù)端相連, 因此客戶端需要指明服務(wù)端對(duì)應(yīng)的IP地址和端口號(hào)
2)需要設(shè)置超時(shí)返回
3)需要設(shè)置循環(huán)等待, 因?yàn)榛镜腟ocket通信都是一來(lái)一回, 這種來(lái)回是通過(guò)阻塞來(lái)完成的。
4)每個(gè)客戶端連入服務(wù)端的時(shí)候, 都具備本身的ID, 類似于HTTP的Session, 這點(diǎn)容易被忽視。在多客戶端連接中, 可以重點(diǎn)關(guān)注。本文提供的代碼也有所提及, 但沒(méi)有深入, 這點(diǎn)留給讀者進(jìn)一步發(fā)掘。
代碼參照/usage目錄下的客戶端測(cè)試代碼, 注意, 先啟動(dòng)服務(wù)端,或者你拿著NetAssis 來(lái)測(cè)試也不錯(cuò).
2 服務(wù)端Socket API要點(diǎn)
1)服務(wù)端一般是被多個(gè)客戶端連接的, 并且這些連接要求服務(wù)端做相似的處理, 因此這里就將這些相似處理, 抽象成一個(gè)SingleTask.java 接口, 具體的業(yè)務(wù)只需要實(shí)現(xiàn)這樣的接口, 就可以并行的處理這些Task.
2)不能無(wú)限制的讓客戶端連入Server, 因此需要設(shè)置上限值
3)啟動(dòng)線程池, 每個(gè)線程針對(duì)一個(gè)具體的客戶端連接
4)注意接收阻塞位置, 需要設(shè)置死循環(huán), 讀不到數(shù)據(jù)將死守著等待(但別耽誤其它線程處理事情)
5)注意服務(wù)端要在死循環(huán)中偵聽(tīng), 這樣保證不錯(cuò)過(guò)任何來(lái)自客戶端的請(qǐng)求。
代碼參照:/usage目錄下的Server端測(cè)試代碼。
代碼中注釋很多,因此這里就不詳細(xì)述說(shuō)。
常見(jiàn)問(wèn)題
1、客戶端Client的時(shí)候, 如果存在網(wǎng)絡(luò)問(wèn)題, 為了避免網(wǎng)絡(luò)問(wèn)題,造成客戶端長(zhǎng)時(shí)間等待, 此時(shí)要設(shè)置一個(gè)TimeOut
clientSocket = new Socket(); //這個(gè)TimeOut是連接等待時(shí)間 clientSocket.connect(tcpAddress, timeOut);
2、當(dāng)客戶端已經(jīng)連接, 每次收到一個(gè)數(shù)據(jù), 客戶端將啟動(dòng)處理, 假如服務(wù)器長(zhǎng)久不發(fā)數(shù)據(jù), 此時(shí)客戶端會(huì)阻塞等待, 為了避免這個(gè)時(shí)候的等待, 可以設(shè)置一個(gè)超時(shí)
clientSocket.setSoTimeout(timeOut);
Java使用socket實(shí)現(xiàn)一個(gè)多線程web服務(wù)器的方法
除了服務(wù)器類,還包括請(qǐng)求類和響應(yīng)類
請(qǐng)求類:獲取客戶的HTTP請(qǐng)求,分析客戶所需要的文件響應(yīng)類:獲得用戶請(qǐng)求后將用戶需要的文件讀出,添加上HTTP應(yīng)答頭。發(fā)送給客戶端。
服務(wù)器處理類
package com.lp.app.webserver;
import java.io.*;
import java.net.*;
//使用Socket創(chuàng)建一個(gè)WEB服務(wù)器,本程序是多線程系統(tǒng)以提高反應(yīng)速度。
class WebServer
{
public static String WEBROOT = "";//默認(rèn)目錄
public static String defaultPage = "index.htm";//默認(rèn)文件
public static void main (String [] args) throws IOException
{
System.out.println ("服務(wù)器啟動(dòng)...\n");
//使用8080端口提供服務(wù)
ServerSocket server = new ServerSocket (8080);
while (true)
{
//阻塞,直到有客戶連接
Socket sk = server.accept ();
System.out.println ("Accepting Connection...\n");
//啟動(dòng)服務(wù)線程
new WebThread (sk).start ();
}
}
}
//使用線程,為多個(gè)客戶端服務(wù)
class WebThread extends Thread
{
private Socket sk;
WebThread (Socket sk)
{
this.sk = sk;
}
//線程體
public void run ()
{
InputStream in = null;
OutputStream out = null;
try{
in = sk.getInputStream();
out = sk.getOutputStream();
//接收來(lái)自客戶端的請(qǐng)求。
Request rq = new Request(in);
//解析客戶請(qǐng)求
String sURL = rq.parse();
System.out.println("sURL="+sURL);
if(sURL.equals("/"))
sURL = WebServer.defaultPage;
Response rp = new Response(out);
rp.Send(sURL);
}
catch (IOException e)
{
System.out.println (e.toString ());
}
finally
{
System.out.println ("關(guān)閉連接...\n");
//最后釋放資源
try{
if (in != null)
in.close ();
if (out != null)
out.close ();
if (sk != null)
sk.close ();
}
catch (IOException e)
{
}
}
}
}
請(qǐng)求類
package com.lp.app.webserver;
import java.io.*;
import java.net.*;
//獲取客戶的HTTP請(qǐng)求,分析客戶所需要的文件
public class Request{
InputStream in = null;
//獲得輸入流。這是客戶的請(qǐng)求數(shù)據(jù)。
public Request(InputStream input){
this.in = input;
}
//解析客戶的請(qǐng)求
public String parse() {
//從Socket讀取一組數(shù)據(jù)
StringBuffer requestStr = new StringBuffer(2048);
int i;
byte[] buffer = new byte[2048];
try {
i = in.read(buffer);
}
catch (IOException e) {
e.printStackTrace();
i = -1;
}
for (int j=0; j<i; j++) {
requestStr.append((char) buffer[j]);
}
System.out.print(requestStr.toString());
return getUri(requestStr.toString());
}
//獲取URI信息字符
private String getUri(String requestString) {
int index1, index2;
index1 = requestString.indexOf(' ');
if (index1 != -1) {
index2 = requestString.indexOf(' ', index1 + 1);
if (index2 > index1)
return requestString.substring(index1 + 1, index2);
}
return null;
}
}
響應(yīng)類
package com.lp.app.webserver;
import java.io.*;
import java.net.*;
//獲得用戶請(qǐng)求后將用戶需要的文件讀出,添加上HTTP應(yīng)答頭。發(fā)送給客戶端。
public class Response{
OutputStream out = null;
//發(fā)送請(qǐng)求的文件 public void Send(String ref) throws IOException {
byte[] bytes = new byte[2048];
FileInputStream fis = null;
try {
//構(gòu)造文件
File file = new File(WebServer.WEBROOT, ref);
if (file.exists()) {
//構(gòu)造輸入文件流
fis = new FileInputStream(file);
int ch = fis.read(bytes, 0, 2048);
//讀取文件
String sBody = new String(bytes,0);
//構(gòu)造輸出信息
String sendMessage = "HTTP/1.1 200 OK\r\n" +
"Content-Type: text/html\r\n" +
"Content-Length: "+ch+"\r\n" +
"\r\n" +sBody;
//輸出文件
out.write(sendMessage.getBytes());
}else {
// 找不到文件
String errorMessage = "HTTP/1.1 404 File Not Found\r\n" +
"Content-Type: text/html\r\n" +
"Content-Length: 23\r\n" +
"\r\n" +
"<h1>File Not Found</h1>";
out.write(errorMessage.getBytes());
}
}
catch (Exception e) {
// 如不能實(shí)例化File對(duì)象,拋出異常。
System.out.println(e.toString() );
}
finally {
if (fis != null)
fis.close();
}
}
//獲取輸出流
public Response(OutputStream output) {
this.out = output;
}
}
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Spring?Data?JPA?實(shí)體類中常用注解說(shuō)明
這篇文章主要介紹了Spring?Data?JPA?實(shí)體類中常用注解說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-11-11
詳解HTTP請(qǐng)求與響應(yīng)基礎(chǔ)及實(shí)例
這篇文章主要介紹了詳解HTTP請(qǐng)求與響應(yīng)基礎(chǔ)及實(shí)例的相關(guān)資料,這里對(duì)http的請(qǐng)求和響應(yīng)進(jìn)行詳細(xì)分析并附有實(shí)現(xiàn)實(shí)例,需要的朋友可以參考下2017-07-07
Java字符串格式化功能?String.format用法詳解
String類的format()方法用于創(chuàng)建格式化的字符串以及連接多個(gè)字符串對(duì)象,熟悉C語(yǔ)言的同學(xué)應(yīng)該記得C語(yǔ)言的sprintf()方法,兩者有類似之處,format()方法有兩種重載形式2024-09-09
Dubbo異步調(diào)用的實(shí)現(xiàn)介紹
dubbo默認(rèn)使用同步的方式調(diào)用。但在有些特殊的場(chǎng)景下,我們可能希望異步調(diào)用dubbo接口,從而避免不必要的等待時(shí)間,這時(shí)候我們就需要用到異步。那么dubbo的異步是如何實(shí)現(xiàn)的呢?下面就來(lái)看看這個(gè)問(wèn)題2022-09-09
Java中@ConfigurationProperties實(shí)現(xiàn)自定義配置綁定問(wèn)題分析
這篇文章主要介紹了@ConfigurationProperties實(shí)現(xiàn)自定義配置綁定問(wèn)題,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-08-08
使用vue3.x+vite+element-ui+vue-router+vuex+axios搭建項(xiàng)目
Springboot應(yīng)用中Mybatis輸出SQL日志的3種方法代碼示例

