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通信都是一來一回, 這種來回是通過阻塞來完成的。
4)每個(gè)客戶端連入服務(wù)端的時(shí)候, 都具備本身的ID, 類似于HTTP的Session, 這點(diǎn)容易被忽視。在多客戶端連接中, 可以重點(diǎn)關(guān)注。本文提供的代碼也有所提及, 但沒有深入, 這點(diǎn)留給讀者進(jìn)一步發(fā)掘。
代碼參照/usage目錄下的客戶端測(cè)試代碼, 注意, 先啟動(dòng)服務(wù)端,或者你拿著NetAssis 來測(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)中偵聽, 這樣保證不錯(cuò)過任何來自客戶端的請(qǐng)求。
代碼參照:/usage目錄下的Server端測(cè)試代碼。
代碼中注釋很多,因此這里就不詳細(xì)述說。
常見問題
1、客戶端Client的時(shí)候, 如果存在網(wǎng)絡(luò)問題, 為了避免網(wǎng)絡(luò)問題,造成客戶端長(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();
//接收來自客戶端的請(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)文章
java集合框架 arrayblockingqueue應(yīng)用分析
ArrayBlockingQueue是一個(gè)由數(shù)組支持的有界阻塞隊(duì)列。此隊(duì)列按 FIFO(先進(jìn)先出)原則對(duì)元素進(jìn)行排序。隊(duì)列的頭部 是在隊(duì)列中存在時(shí)間最長(zhǎng)的元素2012-11-11
Java對(duì)象轉(zhuǎn)JSON三種常用的方法
在Java中可以使用多種方式將對(duì)象轉(zhuǎn)換為JSON字符串,下面這篇文章主要給大家介紹了關(guān)于Java對(duì)象轉(zhuǎn)JSON三種常用的方法,文中通過代碼介紹的非常詳細(xì),需要的朋友可以參考下2024-04-04
springboot2 jackson實(shí)現(xiàn)動(dòng)態(tài)返回類字段方式
這篇文章主要介紹了springboot2 jackson實(shí)現(xiàn)動(dòng)態(tài)返回類字段方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-08-08
SpringBoot處理全局統(tǒng)一異常的實(shí)現(xiàn)
這篇文章主要介紹了SpringBoot處理全局統(tǒng)一異常的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-09-09
SpringBoot實(shí)現(xiàn)動(dòng)態(tài)配置及項(xiàng)目打包部署上線功能
本文講解的是如何使用Spring動(dòng)態(tài)配置文件,實(shí)現(xiàn)不同環(huán)境不同配置,靈活切換配置文件;以及講述了如何使用?Maven?打包,然后上傳至Linux服務(wù)器進(jìn)行部署,對(duì)SpringBoot打包部署上線過程感興趣的朋友一起看看吧2022-10-10
Java并發(fā)編程之CountDownLatch源碼解析
這篇文章主要介紹了Java并發(fā)編程之CountDownLatch源碼解析,文中有非常詳細(xì)的代碼示例,對(duì)正在學(xué)習(xí)java并發(fā)編程的小伙伴們有很好的幫助,需要的朋友可以參考下2021-04-04

