Android網(wǎng)絡(luò)編程Socket
在Android的網(wǎng)絡(luò)通訊中,通常會(huì)使用Socket進(jìn)行設(shè)備間數(shù)的數(shù)據(jù)通訊,使用Http來對(duì)網(wǎng)絡(luò)數(shù)據(jù)進(jìn)行請(qǐng)求。
1、Socket(套接字)
不管是有過Java開發(fā)經(jīng)驗(yàn)還是.NET開發(fā)經(jīng)驗(yàn)的同學(xué)都應(yīng)該對(duì)Socket有或多或少的了解,常見的TCP或者UDP協(xié)議其實(shí)都是基于Socket來實(shí)現(xiàn)的。
Socket是用于描述網(wǎng)絡(luò)上的一個(gè)設(shè)備中的一個(gè)進(jìn)程或者應(yīng)用程序的,Socket由IP地址和端口號(hào)兩部分組成。IP地址用來定位設(shè)備,端口號(hào)用來定位 應(yīng)用程序或者進(jìn)程,比如我們常見的運(yùn)行在80端口上的HTTP協(xié)議。Socket的常見格式為:192.168.1.1:1234。
那么應(yīng)用程序是如何通過Socket來與網(wǎng)絡(luò)中的其他設(shè)備進(jìn)行通訊的呢?通常情況下,Socket通信有兩部分,一部分為監(jiān)聽的Server端,一部分為 主動(dòng)請(qǐng)求連接的Client端。Server端會(huì)一直監(jiān)聽Socket中的端口直到有請(qǐng)求為止,當(dāng)Client端對(duì)該端口進(jìn)行連接請(qǐng)求時(shí),Server端 就給予應(yīng)答并返回一個(gè)Socket對(duì)象,以后在Server端與Client端的數(shù)據(jù)交換就可以使用這個(gè)Socket來進(jìn)行操作了。
2、Android中使用Socket進(jìn)行數(shù)據(jù)交換
ServerSocket
建立服務(wù)端(Server)時(shí),需要使用ServerSocket對(duì)象,這個(gè)對(duì)象會(huì)自動(dòng)對(duì)其構(gòu)造函數(shù)中傳入的端口號(hào)進(jìn)行監(jiān)聽,并在收到連接請(qǐng)求后,使 用ServerSocket.accept()方法返回一個(gè)連接的的Socket對(duì)象。這個(gè)方法并不需要我們像在.NET中那樣使用Start方法,它會(huì) 自動(dòng)進(jìn)行監(jiān)聽的。
Socket
不管建立客戶端(Client)還是在進(jìn)行其他數(shù)據(jù)交換方面的操作時(shí),都需要使用Socket類。Socket類在進(jìn)行初始化時(shí)需要出入Server 端的IP地址和端口號(hào),并返回連接到Server端的一個(gè)Socket對(duì)象,如果是連接失敗,那么將返回異常。同ServerSocket,也是自動(dòng)進(jìn)行 連接請(qǐng)求的。
通過上面兩個(gè)步驟后,Server端和Client端就可以連接起來了,但是僅僅連接起來是沒有任何作用的,數(shù)據(jù)交換才是我們的目的,這時(shí)候就需要用到IO流中的OutputStream類和InputStream類。
OutputStream——可寫流
當(dāng)應(yīng)用程序需要對(duì)流進(jìn)行數(shù)據(jù)寫操作時(shí),可以使用Socket.getOutputStream()方法返回的數(shù)據(jù)流進(jìn)行操作。
InputStream——可讀流
當(dāng)應(yīng)用程序要從流中取出數(shù)據(jù)時(shí),可以使用Socket.getInputStream()方法返回的數(shù)據(jù)流進(jìn)行操作。
看看完整的代碼吧
- View Code
- package LiB.Demo;
- import java.io.BufferedReader;
- import java.io.BufferedWriter;
- import java.io.IOException;
- import java.io.InputStreamReader;
- import java.io.OutputStreamWriter;
- import java.net.ServerSocket;
- import java.net.Socket;
- public class SocketHelper {
- private static ServerSocket serverSocket = null;
- private static Socket client = null;
- private final static int port = 9048;
- private static BufferedReader br= null;
- private static BufferedWriter bw = null;
- /**
- * 創(chuàng)建一個(gè)SocketServer對(duì)象用來建立服務(wù)器
- * @throws IOException
- */
- public static void CreateServer() throws IOException
- {
- serverSocket = new ServerSocket(port,10);
- System.out.println("start listening...");
- }
- /**
- * 創(chuàng)建一個(gè)Socket對(duì)象用來連接SocketServer對(duì)象
- * @param dstName Server對(duì)象的ip地址
- * @return
- * @throws IOException
- */
- public static Socket CreateClient(String dstName) throws IOException
- {
- Socket socket = new Socket(dstName, port);
- //Socket sockets = new Socket("192.168.8.12",port);
- return socket;
- }
- /**
- * 返回一個(gè)已經(jīng)連接到服務(wù)器上的Socket對(duì)象
- * @throws IOException
- */
- public static void GetClinetSocket() throws IOException
- {
- client = serverSocket.accept();
- System.out.println("get a connected client");
- }
- /**
- * 向socket對(duì)象所獲取的流中發(fā)送數(shù)據(jù)
- * @param socket
- * @param msg
- * @throws IOException
- */
- public static void SendMsg(Socket socket , String msg) throws IOException
- {
- bw = new BufferedWriter(new OutputStreamWriter(socket.getOutputStream()));
- bw.write(msg);
- bw.flush();
- bw.close();
- }
- /**
- * 獲取socket對(duì)象流中數(shù)據(jù)
- * @param socket
- * @param msg
- * @return
- * @throws IOException
- */
- public static String ReceiveMsg(Socket socket, String msg) throws IOException
- {
- br = new BufferedReader(new InputStreamReader(socket.getInputStream()));
- String receiveMsg = "Receive msg:"+ br.readLine();
- br.close();
- return receiveMsg;
- }
- /**
- * 釋放socket對(duì)象
- * @throws IOException
- */
- public static void Close() throws IOException
- {
- if(client != null)
- {
- client.close();
- }
- if(serverSocket != null)
- {
- serverSocket.close();
- }
- }
- }
下頁將為您介紹HTTP通訊的內(nèi)容
#p#
3、HTTP通訊
在開始前先簡單介紹下HTTP協(xié)議中的兩種不同的請(qǐng)求方式——GET和POST。GET方式在進(jìn)行數(shù)據(jù)請(qǐng)求時(shí),會(huì)把數(shù)據(jù)附加到URL后面?zhèn)鬟f給服務(wù) 器,比如常見的:http://XXX.XXX.XXX/XX.aspx?id=1,POST方式則是將請(qǐng)求的數(shù)據(jù)放到HTTP請(qǐng)求頭中,作為請(qǐng)求頭的一 部分傳入服務(wù)器。所以,在進(jìn)行HTTP編程前,首先要明確究竟使用的哪種方式進(jìn)行數(shù)據(jù)請(qǐng)求的。
在Android中,可以有兩種方式可以用來進(jìn)行Http編程:1、HttpURLConnection;2、HttpClient。
HttpURLConnection
HttpURLConnection是繼承自URLConnection的一個(gè)抽象類,在HTTP編程時(shí),來自HttpURLConnection的類是所有操作的基礎(chǔ),獲取該對(duì)象的代碼如下:
- View Code
- public HttpURLConnection urlconn= null;
- private void Init() throws IOException
- {
- if (urlStr=="")
- {
- urlStr="http://www.baidu.com";
- }
- URL url = new URL(urlStr);
- //打開一個(gè)URL所指向的Connection對(duì)象
- urlconn = (HttpURLConnection)url.openConnection();
- }View Code
- public HttpURLConnection urlconn= null;
- private void Init() throws IOException
- {
- if (urlStr=="")
- {
- urlStr="http://www.baidu.com";
- }
- URL url = new URL(urlStr);
- //打開一個(gè)URL所指向的Connection對(duì)象
- urlconn = (HttpURLConnection)url.openConnection();
- }
HttpURLConnection對(duì)網(wǎng)絡(luò)資源的請(qǐng)求在默認(rèn)情況下是使用GET方式的,所以當(dāng)使用GET方式時(shí),不需要我們做太多的工作:
- View Code
- public HttpURLConnection urlconn= null;
- private void Init() throws IOException
- {
- if (urlStr=="")
- {
- urlStr="http://www.baidu.com";
- }
- URL url = new URL(urlStr);
- //打開一個(gè)URL所指向的Connection對(duì)象
- urlconn = (HttpURLConnection)url.openConnection();
- }
- /**
- * Http中的get請(qǐng)求,在Url中帶有請(qǐng)求的參數(shù),請(qǐng)求的URL格式通常為:"http://XXX.XXXX.com/xx.aspx?param=value"
- * 在android中默認(rèn)的http請(qǐng)求為get方式
- * @return
- * @throws IOException
- */
- public String HttpGetMethod() throws IOException
- {
- if(urlconn == null)
- {
- Init();
- }
- String result = StreamDeal(urlconn.getInputStream());
- urlconn.disconnect();
- return result;
- }
當(dāng)我們需要使用POST方式時(shí),就需要使用setRequestMethod()來設(shè)置請(qǐng)求方式了。
- View Code
- /**
- * Http中的post請(qǐng)求,不在Url中附加任何參數(shù),這些參數(shù)都會(huì)通過cookie或者session等其他方式以鍵值對(duì)的形式key=value傳送到服務(wù)器上,完成一次請(qǐng)求
- * 請(qǐng)求的URL格式通常為:"http://XXX.XXXX.com/xx.aspx"
- * @param param 請(qǐng)求的鍵名
- * @param value 請(qǐng)求的數(shù)據(jù)值
- * @throws IOException
- */
- public String HttpPostMethod(String key,String value) throws IOException
- {
- if (urlconn==null)
- {
- Init();
- }
- //設(shè)置該URLConnection可讀
- urlconn.setDoInput(true);
- //設(shè)置該URLConnection可寫
- urlconn.setDoOutput(true);
- //使用POST方式來提交數(shù)據(jù)
- urlconn.setRequestMethod("POST");
- //不運(yùn)行緩存
- urlconn.setUseCaches(false);
- //當(dāng)使用POST方式進(jìn)行數(shù)據(jù)請(qǐng)求時(shí),我們可以手動(dòng)執(zhí)行connect動(dòng)作,當(dāng)然,這個(gè)動(dòng)作其實(shí)在getOutputStream()方法中會(huì)默認(rèn)執(zhí)行的
- //上面那些設(shè)置URLConnection屬性的動(dòng)作,一定要在connect動(dòng)作執(zhí)行前,因?yàn)橐坏﹦?dòng)作已經(jīng)執(zhí)行,熟悉設(shè)置就沒有任何作用了
- urlconn.connect();
- //使用POST方式時(shí),我們需要自己構(gòu)造部分Http請(qǐng)求的內(nèi)容,因此我們需要使用OutputStream來進(jìn)行數(shù)據(jù)寫如操作
- OutputStreamWriter writer = new OutputStreamWriter(urlconn.getOutputStream());
- String urlQueryStr = key+"="+URLEncoder.encode(value, "Utf-8");
- writer.write(urlQueryStr);
- writer.flush();
- writer.close();
- //獲取返回的內(nèi)容
- String result = StreamDeal(urlconn.getInputStream());
- return result;
- }
HttpClient
這個(gè)類并不是來自Android的,而是來自org.apache.http。和HttpURLConnection相同,HttpClient也存在GET和POST兩種方式。
HttpGet
在HttpClient中,我們可以非常輕松使用HttpGet對(duì)象來通過GET方式進(jìn)行數(shù)據(jù)請(qǐng)求操作,當(dāng)獲得HttpGet對(duì)象后我們可以 使用HttpClient的execute方法來向我們的服務(wù)器發(fā)送請(qǐng)求。在發(fā)送的GET請(qǐng)求被服務(wù)器相應(yīng)后,會(huì)返回一個(gè)HttpResponse響應(yīng)對(duì) 象,利用這個(gè)響應(yīng)的對(duì)象我們能夠獲得響應(yīng)回來的狀態(tài)碼,如:200、400、401等等。
- View Code
- public String HttpGetMethod()
- {
- String result = "";
- try
- {
- HttpGet httpRequest = new HttpGet(urlStr);
- HttpClient httpClient = new DefaultHttpClient();
- HttpResponse httpResponse = httpClient.execute(httpRequest);
- if(httpResponse.getStatusLine().getStatusCode()==HttpStatus.SC_OK)
- {
- result = EntityUtils.toString(httpResponse.getEntity());
- }
- else
- {
- result = "null";
- }
- return result;
- }
- catch(Exception e)
- {
- return null;
- }
- }
HttpPost
當(dāng)我們使用POST方式時(shí),我們可以使用HttpPost類來進(jìn)行操作。當(dāng)獲取了HttpPost對(duì)象后,我們就需要向這個(gè)請(qǐng)求體傳入鍵值 對(duì),這個(gè)鍵值對(duì)我們可以使用NameValuePair對(duì)象來進(jìn)行構(gòu)造,然后再使用HttpRequest對(duì)象最終構(gòu)造我們的請(qǐng)求體,***使用 HttpClient的execute方法來發(fā)送我們的請(qǐng)求,并在得到響應(yīng)后返回一個(gè)HttpResponse對(duì)象。其他操作和我們?cè)贖ttpGet對(duì)象 中的操作一樣。
- View Code
- public String HttpPostMethod(String key,String value)
- {
- String result = "";
- try
- {
- // HttpPost連接對(duì)象
- HttpPost httpRequest = new HttpPost(urlStr);
- // 使用NameValuePair來保存要傳遞的Post參數(shù)
- List<NameValuePair> params = new ArrayList<NameValuePair>();
- // 添加要傳遞的參數(shù)
- params.add(new BasicNameValuePair(key, value));
- // 設(shè)置字符集
- HttpEntity httpentity = new UrlEncodedFormEntity(params, "Utf-8");
- // 請(qǐng)求httpRequest
- httpRequest.setEntity(httpentity);
- // 取得默認(rèn)的HttpClient
- HttpClient httpclient = new DefaultHttpClient();
- // 取得HttpResponse
- HttpResponse httpResponse = httpclient.execute(httpRequest);
- // HttpStatus.SC_OK表示連接成功
- if (httpResponse.getStatusLine().getStatusCode() == HttpStatus.SC_OK) {
- // 取得返回的字符串
- result = EntityUtils.toString(httpResponse.getEntity());
- return result;
- } else {
- return "null";
- }
- }
- catch(Exception e)
- {
- return null;
- }
- }
三、總結(jié)
可以說Android如果不進(jìn)行與網(wǎng)絡(luò)資源進(jìn)行交互的話,它就和我們當(dāng)初的普通系統(tǒng)沒有任何區(qū)別了,所以網(wǎng)絡(luò)編程對(duì)Android開發(fā)來說有非常特殊的意義。