Android中使用HTTP服務(wù)的用法詳解
在Android中,除了使用Java.NET包下的API訪問HTTP服務(wù)之外,我們還可以換一種途徑去完成工作。Android SDK附帶了Apache的HttpClient API。Apache HttpClient是一個(gè)完善的HTTP客戶端,它提供了對(duì)HTTP協(xié)議的全面支持,可以使用HTTP GET和POST進(jìn)行訪問。下面我們就結(jié)合實(shí)例,介紹一下HttpClient的使用方法。
我們新建一個(gè)http項(xiàng)目,項(xiàng)目結(jié)構(gòu)如圖:
在這個(gè)項(xiàng)目中,我們不需要任何的Activity,所有的操作都在單元測(cè)試類HttpTest.java中完成。
因?yàn)槭褂玫搅藛卧獪y(cè)試,所以在這里先介紹一下如何配置Android中的單元測(cè)試。所有配置信息均在AndroidManifest.xml中完成:
<?xml version="1.0" encoding="utf-8"?> <manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.scott.http" android:versionCode="1" android:versionName="1.0"> <application android:icon="@drawable/icon" android:label="@string/app_name"> <!-- 配置測(cè)試要使用的類庫(kù) --> <uses-library android:name="android.test.runner"/> </application> <!-- 配置測(cè)試設(shè)備的主類和目標(biāo)包 --> <instrumentation android:name="android.test.InstrumentationTestRunner" android:targetPackage="com.scott.http"/> <!-- 訪問HTTP服務(wù)所需的網(wǎng)絡(luò)權(quán)限 --> <uses-permission android:name="android.permission.INTERNET"/> <uses-sdk android:minSdkVersion="8" /> </manifest>
然后,我們的單元測(cè)試類需要繼承android.test.AndroidTestCase類,這個(gè)類本身是繼承junit.framework.TestCase,并提供了getContext()方法,用于獲取Android上下文環(huán)境,這個(gè)設(shè)計(jì)非常有用,因?yàn)楹芏郃ndroid API都是需要Context才能完成的。
現(xiàn)在讓我們來看一下我們的測(cè)試用例,HttpTest.java代碼如下:
package com.scot.http.test; import java.io.ByteArrayOutputStream; import java.io.InputStream; import java.util.ArrayList; import java.util.List; import junit.framework.Assert; import org.apache.http.HttpEntity; import org.apache.http.HttpResponse; import org.apache.http.HttpStatus; import org.apache.http.NameValuePair; import org.apache.http.client.HttpClient; import org.apache.http.client.entity.UrlEncodedFormEntity; import org.apache.http.client.methods.HttpGet; import org.apache.http.client.methods.HttpPost; import org.apache.http.entity.mime.MultipartEntity; import org.apache.http.entity.mime.content.InputStreamBody; import org.apache.http.entity.mime.content.StringBody; import org.apache.http.impl.client.DefaultHttpClient; import org.apache.http.message.BasicNameValuePair; import android.test.AndroidTestCase; public class HttpTest extends AndroidTestCase { private static final String PATH = "http://192.168.1.57:8080/web"; public void testGet() throws Exception { HttpClient client = new DefaultHttpClient(); HttpGet get = new HttpGet(PATH + "/TestServlet?id=1001&name=john&age=60"); HttpResponse response = client.execute(get); if (response.getStatusLine().getStatusCode() == HttpStatus.SC_OK) { InputStream is = response.getEntity().getContent(); String result = inStream2String(is); Assert.assertEquals(result, "GET_SUCCESS"); } } public void testPost() throws Exception { HttpClient client = new DefaultHttpClient(); HttpPost post = new HttpPost(PATH + "/TestServlet"); List<NameValuePair> params = new ArrayList<NameValuePair>(); params.add(new BasicNameValuePair("id", "1001")); params.add(new BasicNameValuePair("name", "john")); params.add(new BasicNameValuePair("age", "60")); HttpEntity formEntity = new UrlEncodedFormEntity(params); post.setEntity(formEntity); HttpResponse response = client.execute(post); if (response.getStatusLine().getStatusCode() == HttpStatus.SC_OK) { InputStream is = response.getEntity().getContent(); String result = inStream2String(is); Assert.assertEquals(result, "POST_SUCCESS"); } } public void testUpload() throws Exception { InputStream is = getContext().getAssets().open("books.xml"); HttpClient client = new DefaultHttpClient(); HttpPost post = new HttpPost(PATH + "/UploadServlet"); InputStreamBody isb = new InputStreamBody(is, "books.xml"); MultipartEntity multipartEntity = new MultipartEntity(); multipartEntity.addPart("file", isb); multipartEntity.addPart("desc", new StringBody("this is description.")); post.setEntity(multipartEntity); HttpResponse response = client.execute(post); if (response.getStatusLine().getStatusCode() == HttpStatus.SC_OK) { is = response.getEntity().getContent(); String result = inStream2String(is); Assert.assertEquals(result, "UPLOAD_SUCCESS"); } } //將輸入流轉(zhuǎn)換成字符串 private String inStream2String(InputStream is) throws Exception { ByteArrayOutputStream baos = new ByteArrayOutputStream(); byte[] buf = new byte[1024]; int len = -1; while ((len = is.read(buf)) != -1) { baos.write(buf, 0, len); } return new String(baos.toByteArray()); } }
因?yàn)榇宋募齻€(gè)測(cè)試用例,所以我將會(huì)逐個(gè)介紹一下。
首先,需要注意的是,我們定位服務(wù)器地址時(shí)使用到了IP,因?yàn)檫@里不能用localhost,服務(wù)端是在windows上運(yùn)行,而本單元測(cè)試運(yùn)行在Android平臺(tái),如果使用localhost就意味著在Android內(nèi)部去訪問服務(wù),可能是訪問不到的,所以必須用IP來定位服務(wù)。
我們先來分析一下testGet測(cè)試用例。我們使用了HttpGet,請(qǐng)求參數(shù)直接附在URL后面,然后由HttpClient執(zhí)行GET請(qǐng)求,如果響應(yīng)成功的話,取得響應(yīng)內(nèi)如輸入流,并轉(zhuǎn)換成字符串,最后判斷是否為GET_SUCCESS。
testGet測(cè)試對(duì)應(yīng)服務(wù)端Servlet代碼如下:
@Override protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { System.out.println("doGet method is called."); String id = request.getParameter("id"); String name = request.getParameter("name"); String age = request.getParameter("age"); System.out.println("id:" + id + ", name:" + name + ", age:" + age); response.getWriter().write("GET_SUCCESS"); }
然后再說testPost測(cè)試用例。我們使用了HttpPost,URL后面并沒有附帶參數(shù)信息,參數(shù)信息被包裝成一個(gè)由NameValuePair類型組成的集合的形式,然后經(jīng)過UrlEncodedFormEntity處理后調(diào)用HttpPost的setEntity方法進(jìn)行參數(shù)設(shè)置,最后由HttpClient執(zhí)行。
testPost測(cè)試對(duì)應(yīng)的服務(wù)端代碼如下:
@Override protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { System.out.println("doPost method is called."); String id = request.getParameter("id"); String name = request.getParameter("name"); String age = request.getParameter("age"); System.out.println("id:" + id + ", name:" + name + ", age:" + age); response.getWriter().write("POST_SUCCESS"); }
上面兩個(gè)是最基本的GET請(qǐng)求和POST請(qǐng)求,參數(shù)都是文本數(shù)據(jù)類型,能滿足普通的需求,不過在有的場(chǎng)合例如我們要用到上傳文件的時(shí)候,就不能使用基本的GET請(qǐng)求和POST請(qǐng)求了,我們要使用多部件的POST請(qǐng)求。下面介紹一下如何使用多部件POST操作上傳一個(gè)文件到服務(wù)端。
由于Android附帶的HttpClient版本暫不支持多部件POST請(qǐng)求,所以我們需要用到一個(gè)HttpMime開源項(xiàng)目,該組件是專門處理與MIME類型有關(guān)的操作。因?yàn)镠ttpMime是包含在HttpComponents 項(xiàng)目中的,所以我們需要去apache官方網(wǎng)站下載HttpComponents,然后把其中的HttpMime.jar包放到項(xiàng)目中去,如圖:
然后,我們觀察testUpload測(cè)試用例,我們用HttpMime提供的InputStreamBody處理文件流參數(shù),用StringBody處理普通文本參數(shù),最后把所有類型參數(shù)都加入到一個(gè)MultipartEntity的實(shí)例中,并將這個(gè)multipartEntity設(shè)置為此次POST請(qǐng)求的參數(shù)實(shí)體,然后執(zhí)行POST請(qǐng)求。服務(wù)端Servlet代碼如下:
package com.scott.web.servlet; import java.io.FileOutputStream; import java.io.IOException; import java.util.Iterator; import java.util.List; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import org.apache.commons.fileupload.FileItem; import org.apache.commons.fileupload.FileItemFactory; import org.apache.commons.fileupload.FileUploadException; import org.apache.commons.fileupload.disk.DiskFileItemFactory; import org.apache.commons.fileupload.servlet.ServletFileUpload; @SuppressWarnings("serial") public class UploadServlet extends HttpServlet { @Override @SuppressWarnings("rawtypes") protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { boolean isMultipart = ServletFileUpload.isMultipartContent(request); if (isMultipart) { FileItemFactory factory = new DiskFileItemFactory(); ServletFileUpload upload = new ServletFileUpload(factory); try { List items = upload.parseRequest(request); Iterator iter = items.iterator(); while (iter.hasNext()) { FileItem item = (FileItem) iter.next(); if (item.isFormField()) { //普通文本信息處理 String paramName = item.getFieldName(); String paramValue = item.getString(); System.out.println(paramName + ":" + paramValue); } else { //上傳文件信息處理 String fileName = item.getName(); byte[] data = item.get(); String filePath = getServletContext().getRealPath("/files") + "/" + fileName; FileOutputStream fos = new FileOutputStream(filePath); fos.write(data); fos.close(); } } } catch (FileUploadException e) { e.printStackTrace(); } } response.getWriter().write("UPLOAD_SUCCESS"); } }
服務(wù)端使用apache開源項(xiàng)目FileUpload進(jìn)行處理,所以我們需要commons-fileupload和commons-io這兩個(gè)項(xiàng)目的jar包,對(duì)服務(wù)端開發(fā)不太熟悉的朋友可以到網(wǎng)上查找一下相關(guān)資料。
介紹完上面的三種不同的情況之后,我們需要考慮一個(gè)問題,在實(shí)際應(yīng)用中,我們不能每次都新建HttpClient,而是應(yīng)該只為整個(gè)應(yīng)用創(chuàng)建一個(gè)HttpClient,并將其用于所有HTTP通信。此外,還應(yīng)該注意在通過一個(gè)HttpClient同時(shí)發(fā)出多個(gè)請(qǐng)求時(shí)可能發(fā)生的多線程問題。針對(duì)這兩個(gè)問題,我們需要改進(jìn)一下我們的項(xiàng)目:
1.擴(kuò)展系統(tǒng)默認(rèn)的Application,并應(yīng)用在項(xiàng)目中。
2.使用HttpClient類庫(kù)提供的ThreadSafeClientManager來創(chuàng)建和管理HttpClient。
改進(jìn)后的項(xiàng)目結(jié)構(gòu)如圖:
其中MyApplication擴(kuò)展了系統(tǒng)的Application,代碼如下:
package com.scott.http; import org.apache.http.HttpVersion; import org.apache.http.client.HttpClient; import org.apache.http.conn.ClientConnectionManager; import org.apache.http.conn.scheme.PlainSocketFactory; import org.apache.http.conn.scheme.Scheme; import org.apache.http.conn.scheme.SchemeRegistry; import org.apache.http.conn.ssl.SSLSocketFactory; import org.apache.http.impl.client.DefaultHttpClient; import org.apache.http.impl.conn.tsccm.ThreadSafeClientConnManager; import org.apache.http.params.BasicHttpParams; import org.apache.http.params.HttpParams; import org.apache.http.params.HttpProtocolParams; import org.apache.http.protocol.HTTP; import android.app.Application; public class MyApplication extends Application { private HttpClient httpClient; @Override public void onCreate() { super.onCreate(); httpClient = this.createHttpClient(); } @Override public void onLowMemory() { super.onLowMemory(); this.shutdownHttpClient(); } @Override public void onTerminate() { super.onTerminate(); this.shutdownHttpClient(); } //創(chuàng)建HttpClient實(shí)例 private HttpClient createHttpClient() { HttpParams params = new BasicHttpParams(); HttpProtocolParams.setVersion(params, HttpVersion.HTTP_1_1); HttpProtocolParams.setContentCharset(params, HTTP.DEFAULT_CONTENT_CHARSET); HttpProtocolParams.setUseExpectContinue(params, true); SchemeRegistry schReg = new SchemeRegistry(); schReg.register(new Scheme("http", PlainSocketFactory.getSocketFactory(), 80)); schReg.register(new Scheme("https", SSLSocketFactory.getSocketFactory(), 443)); ClientConnectionManager connMgr = new ThreadSafeClientConnManager(params, schReg); return new DefaultHttpClient(connMgr, params); } //關(guān)閉連接管理器并釋放資源 private void shutdownHttpClient() { if (httpClient != null && httpClient.getConnectionManager() != null) { httpClient.getConnectionManager().shutdown(); } } //對(duì)外提供HttpClient實(shí)例 public HttpClient getHttpClient() { return httpClient; } }
我們重寫了onCreate()方法,在系統(tǒng)啟動(dòng)時(shí)就創(chuàng)建一個(gè)HttpClient;重寫了onLowMemory()和onTerminate()方法,在內(nèi)存不足和應(yīng)用結(jié)束時(shí)關(guān)閉連接,釋放資源。需要注意的是,當(dāng)實(shí)例化DefaultHttpClient時(shí),傳入一個(gè)由ThreadSafeClientConnManager創(chuàng)建的一個(gè)ClientConnectionManager實(shí)例,負(fù)責(zé)管理HttpClient的HTTP連接。
然后,想要讓我們這個(gè)加強(qiáng)版的“Application”生效,需要在AndroidManifest.xml中做如下配置:
<application android:name=".MyApplication" ...> ... </application>
如果我們沒有配置,系統(tǒng)默認(rèn)會(huì)使用android.app.Application,我們添加了配置,系統(tǒng)就會(huì)使用我們的com.scott.http.MyApplication,然后就可以在context中調(diào)用getApplication()來獲取MyApplication實(shí)例。
有了上面的配置,我們就可以在活動(dòng)中應(yīng)用了,HttpActivity.java代碼如下:
package com.scott.http; import java.io.ByteArrayOutputStream; import java.io.InputStream; import org.apache.http.HttpResponse; import org.apache.http.HttpStatus; import org.apache.http.client.HttpClient; import org.apache.http.client.methods.HttpGet; import android.app.Activity; import android.os.Bundle; import android.view.View; import android.widget.Button; import android.widget.Toast; public class HttpActivity extends Activity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); Button btn = (Button) findViewById(R.id.btn); btn.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { execute(); } }); } private void execute() { try { MyApplication app = (MyApplication) this.getApplication(); //獲取MyApplication實(shí)例 HttpClient client = app.getHttpClient(); //獲取HttpClient實(shí)例 HttpGet get = new HttpGet("http://192.168.1.57:8080/web/TestServlet?id=1001&name=john&age=60"); HttpResponse response = client.execute(get); if (response.getStatusLine().getStatusCode() == HttpStatus.SC_OK) { InputStream is = response.getEntity().getContent(); String result = inStream2String(is); Toast.makeText(this, result, Toast.LENGTH_LONG).show(); } } catch (Exception e) { e.printStackTrace(); } } //將輸入流轉(zhuǎn)換成字符串 private String inStream2String(InputStream is) throws Exception { ByteArrayOutputStream baos = new ByteArrayOutputStream(); byte[] buf = new byte[1024]; int len = -1; while ((len = is.read(buf)) != -1) { baos.write(buf, 0, len); } return new String(baos.toByteArray()); } }
點(diǎn)擊“execute”按鈕,執(zhí)行結(jié)果如下:
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- android之HttpPost&HttpGet使用方法介紹
- Android中發(fā)送Http請(qǐng)求(包括文件上傳、servlet接收)的實(shí)例代碼
- 淺析Android系統(tǒng)中HTTPS通信的實(shí)現(xiàn)
- Android HTTP發(fā)送請(qǐng)求和接收響應(yīng)的實(shí)例代碼
- android平臺(tái)HttpGet、HttpPost請(qǐng)求實(shí)例
- Android HttpURLConnection.getResponseCode()錯(cuò)誤解決方法
- Android下通過httpClient發(fā)送GET和POST請(qǐng)求的實(shí)例代碼
- android實(shí)現(xiàn)http中請(qǐng)求訪問添加cookie的方法
相關(guān)文章
Android多功能時(shí)鐘開發(fā)案例(實(shí)戰(zhàn)篇)
這篇文章主要為大家詳細(xì)介紹了Android多功能時(shí)鐘開發(fā)案例,開發(fā)了時(shí)鐘、鬧鐘、計(jì)時(shí)器和秒表,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-05-05如何利用adb卸載手機(jī)預(yù)裝軟件(系統(tǒng)軟件)
對(duì)于Android手機(jī)通常有很多不必要的預(yù)置軟件,但是又無(wú)法卸載,占用桌面有很難受,所以本次使用adb工具來實(shí)現(xiàn)從電腦命令來卸載或停用軟件,下面這篇文章主要給大家介紹了關(guān)于如何利用adb卸載手機(jī)預(yù)裝軟件(系統(tǒng)軟件)的相關(guān)資料,需要的朋友可以參考下2022-09-09android編程實(shí)現(xiàn)系統(tǒng)圖片剪裁的方法
這篇文章主要介紹了android編程實(shí)現(xiàn)系統(tǒng)圖片剪裁的方法,涉及Android針對(duì)圖片的獲取、修改、保存等操作的相關(guān)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-11-11Android調(diào)用系統(tǒng)裁剪的實(shí)現(xiàn)方法
下面小編就為大家分享一篇Android調(diào)用系統(tǒng)裁剪的實(shí)現(xiàn)方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2018-02-02Android獲取手機(jī)SIM卡運(yùn)營(yíng)商信息的方法
這篇文章主要介紹了Android獲取手機(jī)SIM卡運(yùn)營(yíng)商信息的方法,可獲得手機(jī)的型號(hào)、運(yùn)營(yíng)商信息及系統(tǒng)版本等,需要的朋友可以參考下2014-09-09Android后臺(tái)線程和UI線程通訊實(shí)例
這篇文章主要介紹了Android后臺(tái)線程和UI線程通訊實(shí)例,每一步的要點(diǎn)和步驟都有提及,并配有代碼例子,需要的朋友可以參考下2014-06-06Android使用ViewPager實(shí)現(xiàn)滾動(dòng)廣告
這篇文章主要為大家詳細(xì)介紹了Android使用ViewPager實(shí)現(xiàn)滾動(dòng)廣告,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-11-11Android實(shí)現(xiàn)九宮格手勢(shì)密碼
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)九宮格手勢(shì)密碼,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-06-06kotlin anko頁(yè)面跳轉(zhuǎn)實(shí)現(xiàn)方式,攜帶參數(shù)或flag
這篇文章主要介紹了kotlin anko頁(yè)面跳轉(zhuǎn)實(shí)現(xiàn)方式,攜帶參數(shù)或flag,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-03-03