Android网络编程之使用HTTP訪问网络资源
使用HTTP訪問網(wǎng)絡(luò)資源
? ? ? ?前面介紹了 URLConnection己經(jīng)能夠很方便地與指定網(wǎng)站交換信息,URLConnection另一個(gè)子類:HttpURLConnection,HttpURLConnection 在 LIRLConnection的基礎(chǔ)上做了進(jìn)一步改進(jìn),添加了一些用于操作http資源的便捷方法。
1.使用HttpURLConnection
? ? ? HttpURLConnection繼承了URLConnection,因此也可用于向指定站點(diǎn)發(fā)送GET請求 POST請求。它在URLConnection的基礎(chǔ)上提供了例如以下便捷的方法。
1) Int getResponseCode():獲取server的響應(yīng)代碼。
2) String getResponseMessage():獲取server的響應(yīng)消息。
3) String getRequestMethod():獲取發(fā)送請求的方法。
4) void setRequestMethod(String method):設(shè)置發(fā)送請求的方法。
以下通過個(gè)有用的演示樣例來示范使用HttpURLConnection實(shí)現(xiàn)多線程下載。
1.1實(shí)例:多線程下載
? ? ? ?使用多線程下載文件能夠更快地完畢文件的下載,由于client啟動(dòng)多個(gè)線程進(jìn)行下寒意味著server也須要為該client提供相應(yīng)的服務(wù)。如果server同一時(shí)候最多服務(wù)100個(gè)用戶,server中一條線程相應(yīng)一個(gè)用戶,100條線程在計(jì)算機(jī)內(nèi)并發(fā)運(yùn)行,也就是由CPU劃分史 片輪流運(yùn)行,如果A應(yīng)用使用了 99條線程下載文件,那么相當(dāng)于占用了 99個(gè)用戶的資源自然就擁有了較快的下載速度。
提示:實(shí)際上并非client并發(fā)的下載線程越多,程序的下載速度就越快,由于當(dāng)client開啟太多的并發(fā)線程之后,應(yīng)用程序須要維護(hù)每條線程的開銷、線程同步的開銷,這些開銷反而會(huì)導(dǎo)致下載速度減少.
1.2為了實(shí)現(xiàn)多線程下載,程序可按例如以下步驟進(jìn)行:
? 創(chuàng)建URL對象。
? 獲取指定URL對象所指向資源的大小(由getContentLength()方法實(shí)現(xiàn)),此處用了 HttpURLConnection 類。
? 在本地磁盤上創(chuàng)建一個(gè)與網(wǎng)絡(luò)資源同樣大小的空文件。
? 計(jì)算每條線程應(yīng)該下載網(wǎng)絡(luò)資源的哪個(gè)部分(從哪個(gè)字節(jié)開始,到哪個(gè)字節(jié)結(jié)束,依次創(chuàng)建、啟動(dòng)多條線程來下載網(wǎng)絡(luò)資源的指定部分。
1.2該程序提供的下載工具類代碼例如以下。
package com.jph.net;import java.io.InputStream; import java.io.RandomAccessFile; import java.net.HttpURLConnection; import java.net.URL;/*** Description:* 創(chuàng)建ServerSocket監(jiān)聽的主類* @author jph* Date:2014.08.27*/ public class DownUtil {/**下載資源的路徑**/ private String path;/**下載的文件的保存位置**/ private String targetFile;/**須要使用多少線程下載資源**/ private int threadNum;/**下載的線程對象**/ private DownThread[] threads;/**下載的文件的總大小**/ private int fileSize;public DownUtil(String path, String targetFile, int threadNum){this.path = path;this.threadNum = threadNum;// 初始化threads數(shù)組threads = new DownThread[threadNum];this.targetFile = targetFile;}public void download() throws Exception{URL url = new URL(path);HttpURLConnection conn = (HttpURLConnection) url.openConnection();conn.setConnectTimeout(5 * 1000);conn.setRequestMethod("GET");conn.setRequestProperty("Accept","image/gif, image/jpeg, image/pjpeg, image/pjpeg, "+ "application/x-shockwave-flash, application/xaml+xml, "+ "application/vnd.ms-xpsdocument, application/x-ms-xbap, "+ "application/x-ms-application, application/vnd.ms-excel, "+ "application/vnd.ms-powerpoint, application/msword, */*");conn.setRequestProperty("Accept-Language", "zh-CN");conn.setRequestProperty("Charset", "UTF-8");conn.setRequestProperty("Connection", "Keep-Alive");// 得到文件大小fileSize = conn.getContentLength();conn.disconnect();int currentPartSize = fileSize / threadNum + 1;RandomAccessFile file = new RandomAccessFile(targetFile, "rw");// 設(shè)置本地文件的大小file.setLength(fileSize);file.close();for (int i = 0; i < threadNum; i++){// 計(jì)算每條線程的下載的開始位置int startPos = i * currentPartSize;// 每一個(gè)線程使用一個(gè)RandomAccessFile進(jìn)行下載RandomAccessFile currentPart = new RandomAccessFile(targetFile,"rw");// 定位該線程的下載位置currentPart.seek(startPos);// 創(chuàng)建下載線程threads[i] = new DownThread(startPos, currentPartSize,currentPart);// 啟動(dòng)下載線程threads[i].start();}}// 獲取下載的完畢百分比public double getCompleteRate(){// 統(tǒng)計(jì)多條線程已經(jīng)下載的總大小int sumSize = 0;for (int i = 0; i < threadNum; i++){sumSize += threads[i].length;}// 返回已經(jīng)完畢的百分比return sumSize * 1.0 / fileSize;}private class DownThread extends Thread{/**當(dāng)前線程的下載位置**/ private int startPos;/**定義當(dāng)前線程負(fù)責(zé)下載的文件大小**/ private int currentPartSize;/**當(dāng)前線程須要下載的文件塊**/ private RandomAccessFile currentPart;/**定義該線程已下載的字節(jié)數(shù)**/ public int length;public DownThread(int startPos, int currentPartSize,RandomAccessFile currentPart){this.startPos = startPos;this.currentPartSize = currentPartSize;this.currentPart = currentPart;}@Overridepublic void run(){try{URL url = new URL(path);HttpURLConnection conn = (HttpURLConnection)url.openConnection();conn.setConnectTimeout(5 * 1000);conn.setRequestMethod("GET");conn.setRequestProperty("Accept","image/gif, image/jpeg, image/pjpeg, image/pjpeg, "+ "application/x-shockwave-flash, application/xaml+xml, "+ "application/vnd.ms-xpsdocument, application/x-ms-xbap, "+ "application/x-ms-application, application/vnd.ms-excel, "+ "application/vnd.ms-powerpoint, application/msword, */*");conn.setRequestProperty("Accept-Language", "zh-CN");conn.setRequestProperty("Charset", "UTF-8");InputStream inStream = conn.getInputStream();// 跳過startPos個(gè)字節(jié),表明該線程僅僅下載自己負(fù)責(zé)哪部分文件。inStream.skip(this.startPos);byte[] buffer = new byte[1024];int hasRead = 0;// 讀取網(wǎng)絡(luò)數(shù)據(jù),并寫入本地文件while (length < currentPartSize&& (hasRead = inStream.read(buffer)) > 0){currentPart.write(buffer, 0, hasRead);// 累計(jì)該線程下載的總大小length += hasRead;}currentPart.close();inStream.close();}catch (Exception e){e.printStackTrace();}}} }? ? ?上而的DownUtil工具類中包含一個(gè)DownloadThread內(nèi)部類,該內(nèi)部類的run()方法中負(fù)責(zé)打開遠(yuǎn)程資源的輸入流,并調(diào)用inputStream的skip(int)方法跳過指定數(shù)量的字節(jié),這樣就讓該線程讀取由它自己負(fù)責(zé)下載的部分。
? ? ? 提供了上面的DownUtil工具類之后,接下來就能夠在Activity中調(diào)用該DownUtil類來運(yùn)行下載任務(wù),該程序界面中包括兩個(gè)文本框,一個(gè)用于輸入網(wǎng)絡(luò)文件的源路徑,還有一個(gè)用于指定下載到本地的文件的文件名稱,該程序的界面比較簡單,故此處不再給出界面布局代碼。該程序的Activity代碼例如以下。
package com.jph.net;import java.util.Timer; import java.util.TimerTask; import android.app.Activity; import android.content.Context; import android.os.Bundle; import android.os.Handler; import android.os.Looper; import android.os.Message; import android.view.View; import android.view.View.OnClickListener; import android.widget.Button; import android.widget.EditText; import android.widget.ProgressBar; import android.widget.Toast;/*** Description:* 多線程下載* @author jph* Date:2014.08.27*/ public class MultiThreadDown extends Activity {EditText url;EditText target;Button downBn;ProgressBar bar;DownUtil downUtil;private int mDownStatus;@Overridepublic void onCreate(Bundle savedInstanceState){super.onCreate(savedInstanceState);setContentView(R.layout.main);// 獲取程序界面中的三個(gè)界面控件url = (EditText) findViewById(R.id.url);target = (EditText) findViewById(R.id.target);downBn = (Button) findViewById(R.id.down);bar = (ProgressBar) findViewById(R.id.bar);// 創(chuàng)建一個(gè)Handler對象final Handler handler = new Handler(){@Overridepublic void handleMessage(Message msg){if (msg.what == 0x123){bar.setProgress(mDownStatus);}}};downBn.setOnClickListener(new OnClickListener(){@Overridepublic void onClick(View v){// 初始化DownUtil對象(最后一個(gè)參數(shù)指定線程數(shù))downUtil = new DownUtil(url.getText().toString(),target.getText().toString(), 6);new Thread(){@Overridepublic void run(){try{// 開始下載downUtil.download();}catch (Exception e){e.printStackTrace();}// 定義每秒調(diào)度獲取一次系統(tǒng)的完畢進(jìn)度final Timer timer = new Timer();timer.schedule(new TimerTask(){@Overridepublic void run(){// 獲取下載任務(wù)的完畢比率double completeRate = downUtil.getCompleteRate();mDownStatus = (int) (completeRate * 100);// 發(fā)送消息通知界面更新進(jìn)度條handler.sendEmptyMessage(0x123);// 下載全然后取消任務(wù)調(diào)度if (mDownStatus >= 100){showToastByRunnable(MultiThreadDown.this, "下載完畢", 2000);timer.cancel();}}}, 0, 100);}}.start();}});}/*** 在非UI線程中使用Toast* @param context 上下文* @param text 用以顯示的消息內(nèi)容* @param duration 消息顯示的時(shí)間* */private void showToastByRunnable(final Context context, final CharSequence text, final int duration) {Handler handler = new Handler(Looper.getMainLooper());handler.post(new Runnable() {@Overridepublic void run() {Toast.makeText(context, text, duration).show();}});} }
? ? ? ?上面的Activity不僅使用了 DownUtil來控制程序下載,并且程序還啟動(dòng)了一個(gè)定時(shí)器,該定時(shí)器控制每隔0.1秒査詢一次下載進(jìn)度,并通過程序中的進(jìn)度條來顯示任務(wù)的下載進(jìn)度。
? ? ? ?該程序不僅須要訪問網(wǎng)絡(luò),還須要訪問系統(tǒng)SD卡,在SD卡中創(chuàng)建文件,因此必須授予該程序訪問網(wǎng)絡(luò)、訪問SD卡文件的權(quán)限:
<!-- 在SD卡中創(chuàng)建與刪除文件權(quán)限 --> <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/> <!-- 向SD卡寫入數(shù)據(jù)權(quán)限 --> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/> <!-- 授權(quán)訪問網(wǎng)絡(luò) --> <uses-permission android:name="android.permission.INTERNET"/>
程序執(zhí)行效果圖:
?
提示:上面的程序已經(jīng)實(shí)現(xiàn)了多線程下載的核心代碼,假設(shè)要實(shí)現(xiàn)斷點(diǎn)下載,則還須要額外添加一個(gè)配置文件(大家能夠發(fā)現(xiàn)全部斷點(diǎn)下載工具都會(huì)在下載開始生成兩個(gè)文件:一個(gè)是與網(wǎng)絡(luò)資源同樣大小的空文件,一個(gè)是配置文件),該配置文件分別記錄每一個(gè)線程已經(jīng)下載到了哪個(gè)字節(jié),當(dāng)網(wǎng)絡(luò)斷開后再次開始下載時(shí),每一個(gè)線程依據(jù)配置文件中記錄的位置向后下載就可以。????
2 使用ApacheHttpClient
? ? ? ? 在普通情況下,假設(shè)僅僅是須要向Web網(wǎng)站的某個(gè)簡單頁面提交請求并獲取server響應(yīng), 全然能夠使用前面所介紹的HttpURLConnection來完畢。但在絕大部分情況下,Web網(wǎng)站的網(wǎng)頁可能沒這么簡單,這些頁面并非通過一個(gè)簡單的URL就可訪問的,可能須要用戶登錄并且具有對應(yīng)的權(quán)限才可訪問該頁面。在這樣的情況下,就須要涉及Session、Cookie的處理了,假設(shè)打算使用HttpURLConnection來處理這些細(xì)節(jié),當(dāng)然也是可能實(shí)現(xiàn)的,僅僅是處理起來難度就大了。
? ? ? ? 為了更好地處理向Web網(wǎng)站請求,包含處理Session、Cookie等細(xì)節(jié)問題,Apache開源組織提供了一個(gè)HttpClient項(xiàng)目,看它的名稱就知道,它是一個(gè)簡單的HTTPclient(并非瀏覽器),能夠用于發(fā)送HTTP請求,接收HTTP響應(yīng)。但不會(huì)緩存server的響應(yīng),不能運(yùn)行HTML頁面中嵌入的JavaScript代碼;也不會(huì)對頁面內(nèi)容進(jìn)行不論什么解析、處理。
提示:簡單來說,HttpClient就是一個(gè)增強(qiáng)版的HttpURLConnection ,HttpURLConnection能夠做的事情 HttpClient所有能夠做;HttpURLConnection沒有提供的有些功能,HttpClient也提供了,但它僅僅是關(guān)注于怎樣發(fā)送請求、接收響應(yīng),以及管理HTTP連接。|
Android集成了HttpClient,開發(fā)者能夠直接在Android應(yīng)用中使用HttpCHent來訪問提交請求、接收響應(yīng)。
2.1使用HttpClient發(fā)送清求、接收響應(yīng)非常easy,僅僅要例如以下幾步就可以:
1) 創(chuàng)建HttpClient對象。
2) 假設(shè)須要發(fā)送GET請求,創(chuàng)建HttpGet對象;假設(shè)須要發(fā)送POST 求,創(chuàng)建HttpPost對象。
3) 假設(shè)須要發(fā)送請求參數(shù),可調(diào)用HttpGet、HttpPost共同的setParams(HttpParams params)方法來加入請求參數(shù);對于HttpPost對象而言,也可調(diào)用setEntity(HttpEntityentity)方法來設(shè)置請求參數(shù)。
4) 調(diào)用HttpClient對象的execute(HttpUriRequestrequest)發(fā)送請求,運(yùn)行該方法返回一 個(gè) HttpResponse。
5) 調(diào)用HttpResponse的getAllHeaders()、getHeaders(String name)等方法可獲取server的響應(yīng)頭;調(diào)用HttpResponse的getEntity()方法可獲取HttpEntity對象,該對象包裝了server的響應(yīng)內(nèi)容。程序可通過該對象獲取server的響應(yīng)內(nèi)容。
2.2實(shí)例:訪問被保護(hù)資源
?? ? 以下的Android應(yīng)用須要向指定頁面發(fā)送請求,但該頁面并非一個(gè)簡單的頁面,僅僅有 當(dāng)用戶已經(jīng)登錄,并且登錄用戶的username是jph時(shí)才可訪問該頁面。假設(shè)使用HttpUrlConnection來訪問該頁面,那么須要處理的細(xì)節(jié)就太復(fù)雜了。以下將會(huì)借助于 HttpClient來訪問被保護(hù)的頁面。
? ? ? 訪問Web應(yīng)用中被保護(hù)的頁面,假設(shè)使用瀏覽器則十分簡單,用戶通過系統(tǒng)提供的登錄頁面登錄系統(tǒng),瀏覽器會(huì)負(fù)責(zé)維護(hù)與server之間的Session,假設(shè)用戶登錄的username、password符合要求,就能夠訪問被保護(hù)資源了。
? ? ? 為了通過HttpClient來訪問被保護(hù)頁面,程序相同須要使用HttpClient來登錄系統(tǒng),僅僅要應(yīng)用程序使用同一個(gè)HttpClient發(fā)送請求,HttpClient會(huì)自己主動(dòng)維護(hù)與server之間的Session狀態(tài),也就是說程序第一次使用HttpCHent登錄系統(tǒng)后,接下來使用HttpCHent就可以訪問被保護(hù)頁面了。
提示:盡管此處給出的實(shí)例僅僅是訪問被保護(hù)的頁面,但訪問其它被保護(hù)的資源也與此類似,程序僅僅要第一次通過HttpClient登錄系統(tǒng),接下來就可以通過該HttpClient訪問被保護(hù)資源了。
2.3程序代碼:
??
package com.jph.net;import java.io.BufferedReader; import java.io.InputStreamReader; import java.util.ArrayList; import java.util.List; import org.apache.http.HttpEntity; import org.apache.http.HttpResponse; 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.impl.client.DefaultHttpClient; import org.apache.http.message.BasicNameValuePair; import org.apache.http.protocol.HTTP; import org.apache.http.util.EntityUtils; import android.app.Activity; import android.app.AlertDialog; import android.content.DialogInterface; import android.os.Bundle; import android.os.Handler; import android.os.Looper; import android.os.Message; import android.text.Html; import android.view.View; import android.widget.EditText; import android.widget.TextView; import android.widget.Toast; /*** Description:* 使用HttpClient訪問受保護(hù)的網(wǎng)絡(luò)資源* @author jph* Date:2014.08.28*/ public class HttpClientDemo extends Activity {TextView response;HttpClient httpClient;Handler handler = new Handler(){public void handleMessage(Message msg){if(msg.what == 0x123){// 使用response文本框顯示server響應(yīng)response.append(Html.fromHtml(msg.obj.toString()) + "\n");}}};@Overridepublic void onCreate(Bundle savedInstanceState){super.onCreate(savedInstanceState);setContentView(R.layout.main);// 創(chuàng)建DefaultHttpClient對象httpClient = new DefaultHttpClient();response = (TextView) findViewById(R.id.response);}/*** 此方法用于響應(yīng)“訪問頁面”button* */public void accessSecret(View v){response.setText("");new Thread(){@Overridepublic void run(){// 創(chuàng)建一個(gè)HttpGet對象HttpGet get = new HttpGet("http://10.201.1.32:8080/HttpClientTest_Web/secret.jsp"); //①try{// 發(fā)送GET請求HttpResponse httpResponse = httpClient.execute(get);//②HttpEntity entity = httpResponse.getEntity();if (entity != null){// 讀取server響應(yīng)BufferedReader br = new BufferedReader(new InputStreamReader(entity.getContent()));String line = null;while ((line = br.readLine()) != null){Message msg = new Message();msg.what = 0x123;msg.obj = line;handler.sendMessage(msg);}}}catch (Exception e){e.printStackTrace();}}}.start();}/*** 此方法用于響應(yīng)“登陸系統(tǒng)”button* */public void showLogin(View v){// 載入登錄界面final View loginDialog = getLayoutInflater().inflate(R.layout.login, null);// 使用對話框供用戶登錄系統(tǒng)new AlertDialog.Builder(HttpClientDemo.this).setTitle("登錄系統(tǒng)").setView(loginDialog).setPositiveButton("登錄",new DialogInterface.OnClickListener(){@Overridepublic void onClick(DialogInterface dialog,int which){// 獲取用戶輸入的username、passwordfinal String name = ((EditText) loginDialog.findViewById(R.id.name)).getText().toString();final String pass = ((EditText) loginDialog.findViewById(R.id.pass)).getText().toString();new Thread(){@Overridepublic void run(){try{HttpPost post = new HttpPost("http://10.201.1.32:8080/" +"HttpClientTest_Web/login.jsp");//③// 假設(shè)傳遞參數(shù)個(gè)數(shù)比較多的話能夠?qū)鬟f的參數(shù)進(jìn)行封裝List<NameValuePair> params = newArrayList<NameValuePair>();params.add(new BasicNameValuePair("name", name));params.add(new BasicNameValuePair("pass", pass)); // 設(shè)置請求參數(shù)post.setEntity(new UrlEncodedFormEntity(params, HTTP.UTF_8));// 發(fā)送POST請求HttpResponse response = httpClient.execute(post); //④// 假設(shè)server成功地返回響應(yīng)if (response.getStatusLine().getStatusCode() == 200){String msg = EntityUtils.toString(response.getEntity());Looper.prepare();// 提示登錄成功Toast.makeText(HttpClientDemo.this,msg, Toast.LENGTH_SHORT).show();Looper.loop();}}catch (Exception e){e.printStackTrace();}}}.start();}}).setNegativeButton("取消", null).show();} }? ? ? ? ?上面的程序中①、②號(hào)粗體字代碼先創(chuàng)建了一個(gè)HttpGet對象,接下來程序調(diào)用HttpClient的execute()方法發(fā)送GET請求;程序中③、④號(hào)粗體字代碼先創(chuàng)建了一個(gè)HttpPost對象,接下來程序調(diào)用了HttpClient的execute()方法發(fā)送POST請求。上面的GET請求用于獲取server上的被保護(hù)頁面,POST請求用于登錄系統(tǒng)。
執(zhí)行該程序,單擊“訪問頁面”button將可看到例如以下圖所看到的的頁面。
? ? ? ? 從上圖能夠看出,程序直接向指定Web應(yīng)用的被保護(hù)頁面secret.jsp發(fā)送請求,程序?qū)o法訪問被保護(hù)頁面,于是看到下圖所看到的的頁面。單擊下圖所看到的頁面中的“登錄”button,系統(tǒng)將會(huì)顯演示樣例如以下圖所看到的的登錄對話框。
? ? ? ?在上圖所看到的對話框的兩個(gè)輸入框中分別輸入“jph”、“123”,然后單擊“登錄”button,系統(tǒng)將會(huì)向Web網(wǎng)站的login.jsp頁面發(fā)送POST請求,并將用戶輸入的username、password作為請求參數(shù)。假設(shè)username、password正確,就可以看到登錄成功的提示。
? ? ? ? 登錄成功后,HttpClient將會(huì)自己主動(dòng)維護(hù)與server之間的連接,并維護(hù)與server之間的Session狀態(tài),再次單擊程序中的“訪問頁面”button,就可以看到例如以下圖所看到的的輸出。
? ? ? ? 從上圖能夠看出,此時(shí)使用HttpClient發(fā)送GET請求就可以正常訪問被保護(hù)資源,這就是由于前面使用了HttpClient登錄了系統(tǒng),并且HttpClient能夠維護(hù)與server之間的Session連接。
從上面的編程過程不難看出,使用Apache的HttpClient更加簡單,并且它比HttpURLConnection提供了很多其它的功能。
?
總結(jié)
以上是生活随笔為你收集整理的Android网络编程之使用HTTP訪问网络资源的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Sticker.js生成图片或者页面元素
- 下一篇: 初探swift语言的学习笔记五(线程)