欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Android編程中HTTP服務(wù)用法詳解

 更新時(shí)間:2016年01月05日 11:29:21   作者:liuhe688  
這篇文章主要介紹了Android編程中HTTP服務(wù)用法,結(jié)合實(shí)例形式較為詳細(xì)的分析了Android中http服務(wù)的具體步驟與相關(guān)使用技巧,需要的朋友可以參考下

本文實(shí)例講述了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,所有的操作都在單元測試類HttpTest.java中完成。

因?yàn)槭褂玫搅藛卧獪y試,所以在這里先介紹一下如何配置Android中的單元測試。所有配置信息均在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">
    <!-- 配置測試要使用的類庫 -->
    <uses-library android:name="android.test.runner"/>
  </application>
  <!-- 配置測試設(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>

然后,我們的單元測試類需要繼承android.test.AndroidTestCase類,這個(gè)類本身是繼承junit.framework.TestCase,并提供了getContext()方法,用于獲取Android上下文環(huán)境,這個(gè)設(shè)計(jì)非常有用,因?yàn)楹芏郃ndroid API都是需要Context才能完成的。

現(xiàn)在讓我們來看一下我們的測試用例,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è)測試用例,所以我將會(huì)逐個(gè)介紹一下。

首先,需要注意的是,我們定位服務(wù)器地址時(shí)使用到了IP,因?yàn)檫@里不能用localhost,服務(wù)端是在windows上運(yùn)行,而本單元測試運(yùn)行在Android平臺(tái),如果使用localhost就意味著在Android內(nèi)部去訪問服務(wù),可能是訪問不到的,所以必須用IP來定位服務(wù)。

我們先來分析一下testGet測試用例。我們使用了HttpGet,請求參數(shù)直接附在URL后面,然后由HttpClient執(zhí)行GET請求,如果響應(yīng)成功的話,取得響應(yīng)內(nèi)如輸入流,并轉(zhuǎn)換成字符串,最后判斷是否為GET_SUCCESS。

testGet測試對(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測試用例。我們使用了HttpPost,URL后面并沒有附帶參數(shù)信息,參數(shù)信息被包裝成一個(gè)由NameValuePair類型組成的集合的形式,然后經(jīng)過UrlEncodedFormEntity處理后調(diào)用HttpPost的setEntity方法進(jìn)行參數(shù)設(shè)置,最后由HttpClient執(zhí)行。

testPost測試對(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請求和POST請求,參數(shù)都是文本數(shù)據(jù)類型,能滿足普通的需求,不過在有的場合例如我們要用到上傳文件的時(shí)候,就不能使用基本的GET請求和POST請求了,我們要使用多部件的POST請求。下面介紹一下如何使用多部件POST操作上傳一個(gè)文件到服務(wù)端。

由于Android附帶的HttpClient版本暫不支持多部件POST請求,所以我們需要用到一個(gè)HttpMime開源項(xiàng)目,該組件是專門處理與MIME類型有關(guān)的操作。因?yàn)镠ttpMime是包含在HttpComponents 項(xiàng)目中的,所以我們需要去apache官方網(wǎng)站下載HttpComponents,然后把其中的HttpMime.jar包放到項(xiàng)目中去,如圖:

然后,我們觀察testUpload測試用例,我們用HttpMime提供的InputStreamBody處理文件流參數(shù),用StringBody處理普通文本參數(shù),最后把所有類型參數(shù)都加入到一個(gè)MultipartEntity的實(shí)例中,并將這個(gè)multipartEntity設(shè)置為此次POST請求的參數(shù)實(shí)體,然后執(zhí)行POST請求。服務(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è)請求時(shí)可能發(fā)生的多線程問題。針對(duì)這兩個(gè)問題,我們需要改進(jìn)一下我們的項(xiàng)目:

1.擴(kuò)展系統(tǒng)默認(rèn)的Application,并應(yīng)用在項(xiàng)目中。

2.使用HttpClient類庫提供的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é)果如下:

希望本文所述對(duì)大家Android程序設(shè)計(jì)有所幫助。

相關(guān)文章

最新評(píng)論