使用HTTP访问网络资源
前面介绍了 URLConnection己经可以非常方便地与指定站点交换信息,URLConnection还有一个子类:HttpURLConnection,HttpURLConnection 在 LIRLConnection的基础上做了进一步改进,增加了一些用于操作http资源的便捷方法。
1.使用HttpURLConnection
HttpURLConnection继承了URLConnection,因此也可用于向指定网站发送GET请求 POST请求。它在URLConnection的基础上提供了如下便捷的方法。
1) Int getResponseCode():获取服务器的响应代码。
2) String getResponseMessage():获取服务器的响应消息。
3) String getRequestMethod():获取发送请求的方法。
4) void setRequestMethod(String method):设置发送请求的方法。
下面通过个实用的示例来示范使用HttpURLConnection实现多线程下载。
1.1实例:多线程下载
使用多线程下载文件可以更快地完成文件的下载,因为客户端启动多个线程进行下寒意味着服务器也需要为该客户端提供相应的服务。假设服务器同时最多服务100个用户,服务器中一条线程对应一个用户,100条线程在计算机内并发执行,也就是由CPU划分史 片轮流执行,如果A应用使用了 99条线程下载文件,那么相当于占用了 99个用户的资源自然就拥有了较快的下载速度。
提示:实际上并不是客户端并发的下载线程越多,程序的下载速度就越快,因为当客户端开启太多的并发线程之后,应用程序需要维护每条线程的开销、线程同步的开销,这些开销反而会导致下载速度降低.
1.2为了实现多线程下载,程序可按如下步骤进行:
Ø 创建URL对象。
Ø 获取指定URL对象所指向资源的大小(由getContentLength()方法实现),此处用了 HttpURLConnection 类。
Ø 在本地磁盘上创建一个与网络资源相同大小的空文件。
Ø 计算每条线程应该下载网络资源的哪个部分(从哪个字节开始,到哪个字节结束,依次创建、启动多条线程来下载网络资源的指定部分。
1.2该程序提供的下载工具类代码如下。
- package com.jph.net;
-
- import java.io.InputStream;
- import java.io.RandomAccessFile;
- import java.net.HttpURLConnection;
- import java.net.URL;
-
-
-
-
-
-
-
- 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 = 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");
-
- file.setLength(fileSize);
- file.close();
- for (int i = 0; i < threadNum; i++)
- {
-
- int startPos = i * currentPartSize;
-
- RandomAccessFile currentPart = new RandomAccessFile(targetFile,
- "rw");
-
- currentPart.seek(startPos);
-
- threads[i] = new DownThread(startPos, currentPartSize,
- currentPart);
-
- threads[i].start();
- }
- }
-
-
- public double getCompleteRate()
- {
-
- int sumSize = 0;
- for (int i = 0; i < threadNum; i++)
- {
- sumSize += threads[i].length;
- }
-
- return sumSize * 1.0 / fileSize;
- }
-
- private class DownThread extends Thread
- {
- /**当前线程的下载位置**/
- private int startPos;
-
- private int currentPartSize;
-
- private RandomAccessFile currentPart;
-
- public int length;
-
- public DownThread(int startPos, int currentPartSize,
- RandomAccessFile currentPart)
- {
- this.startPos = startPos;
- this.currentPartSize = currentPartSize;
- this.currentPart = currentPart;
- }
-
- @Override
- public 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();
-
- inStream.skip(this.startPos);
- byte[] buffer = new byte[1024];
- int hasRead = 0;
-
- while (length < currentPartSize
- && (hasRead = inStream.read(buffer)) > 0)
- {
- currentPart.write(buffer, 0, hasRead);
-
- length += hasRead;
- }
- currentPart.close();
- inStream.close();
- }
- catch (Exception e)
- {
- e.printStackTrace();
- }
- }
- }
- }
上而的DownUtil工具类中包括一个DownloadThread内部类,该内部类的run()方法中负责打开远程资源的输入流,并调用inputStream的skip(int)方法跳过指定数量的字节,这样就让该线程读取由它自己负责下载的部分。
提供了上面的DownUtil工具类之后,接下来就可以在Activity中调用该DownUtil类来执行下载任务,该程序界面中包含两个文本框,一个用于输入网络文件的源路径,另一个用于指定下载到本地的文件的文件名,该程序的界面比较简单,故此处不再给出界面布局代码。该程序的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;
-
-
-
-
-
-
-
- public class MultiThreadDown extends Activity
- {
- EditText url;
- EditText target;
- Button downBn;
- ProgressBar bar;
- DownUtil downUtil;
- private int mDownStatus;
-
- @Override
- public void onCreate(Bundle savedInstanceState)
- {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.main);
-
- url = (EditText) findViewById(R.id.url);
- target = (EditText) findViewById(R.id.target);
- downBn = (Button) findViewById(R.id.down);
- bar = (ProgressBar) findViewById(R.id.bar);
-
- final Handler handler = new Handler()
- {
- @Override
- public void handleMessage(Message msg)
- {
- if (msg.what == 0x123)
- {
- bar.setProgress(mDownStatus);
- }
- }
- };
- downBn.setOnClickListener(new OnClickListener()
- {
- @Override
- public void onClick(View v)
- {
-
- downUtil = new DownUtil(url.getText().toString(),
- target.getText().toString(), 6);
- new Thread()
- {
- @Override
- public void run()
- {
- try
- {
-
- downUtil.download();
- }
- catch (Exception e)
- {
- e.printStackTrace();
- }
-
- final Timer timer = new Timer();
- timer.schedule(new TimerTask()
- {
- @Override
- public void run()
- {
-
- double completeRate = downUtil.getCompleteRate();
- mDownStatus = (int) (completeRate * 100);
-
- handler.sendEmptyMessage(0x123);
-
- if (mDownStatus >= 100)
- {
- showToastByRunnable(MultiThreadDown.this, "下载完成", 2000);
- timer.cancel();
- }
- }
- }, 0, 100);
- }
- }.start();
- }
- });
- }
-
-
-
-
-
-
- private void showToastByRunnable(final Context context, final CharSequence text, final int duration) {
- Handler handler = new Handler(Looper.getMainLooper());
- handler.post(new Runnable() {
- @Override
- public void run() {
- Toast.makeText(context, text, duration).show();
- }
- });
- }
- }
上面的Activity不仅使用了 DownUtil来控制程序下载,而且程序还启动了一个定时器,该定时器控制每隔0.1秒査询一次下载进度,并通过程序中的进度条来显示任务的下载进度。
该程序不仅需要访问网络,还需要访问系统SD卡,在SD卡中创建文件,因此必须授予该程序访问网络、访问SD卡文件的权限:
-
- <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/>
-
- <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
-
- <uses-permission android:name="android.permission.INTERNET"/>
程序运行效果图:
提示:上面的程序已经实现了多线程下载的核心代码,如果要实现断点下载,则还需要额外增加一个配置文件(大家可以发现所有断点下载工具都会在下载开始生成两个文件:一个是与网络资源相同大小的空文件,一个是配置文件),该配置文件分别记录每个线程已经下载到了哪个字节,当网络断开后再次开始下载时,每个线程根据配置文件里记录的位置向后下载即可。
2 使用ApacheHttpClient
在一般情况下,如果只是需要向Web站点的某个简单页面提交请求并获取服务器响应, 完全可以使用前面所介绍的HttpURLConnection来完成。但在绝大部分情况下,Web站点的网页可能没这么简单,这些页面并不是通过一个简单的URL就可访问的,可能需要用户登录而且具有相应的权限才可访问该页面。在这种情况下,就需要涉及Session、Cookie的处理了,如果打算使用HttpURLConnection来处理这些细节,当然也是可能实现的,只是处理起来难度就大了。
为了更好地处理向Web站点请求,包括处理Session、Cookie等细节问题,Apache开源组织提供了一个HttpClient项目,看它的名称就知道,它是一个简单的HTTP客户端(并不是浏览器),可以用于发送HTTP请求,接收HTTP响应。但不会缓存服务器的响应,不能执行HTML页面中嵌入的JavaScript代码;也不会对页面内容进行任何解析、处理。
提示:简单来说,HttpClient就是一个增强版的HttpURLConnection ,HttpURLConnection可以做的事情 HttpClient全部可以做;HttpURLConnection没有提供的有些功能,HttpClient也提供了,但它只是关注于如何发送请求、接收响应,以及管理HTTP连接。|
Android集成了HttpClient,开发人员可以直接在Android应用中使用HttpCHent来访问提交请求、接收响应。
2.1使用HttpClient发送清求、接收响应很简单,只要如下几步即可:
1) 创建HttpClient对象。
2) 如果需要发送GET请求,创建HttpGet对象;如果需要发送POST 求,创建HttpPost对象。
3) 如果需要发送请求参数,可调用HttpGet、HttpPost共同的setParams(HttpParams params)方法来添加请求参数;对于HttpPost对象而言,也可调用setEntity(HttpEntityentity)方法来设置请求参数。
4) 调用HttpClient对象的execute(HttpUriRequestrequest)发送请求,执行该方法返回一 个 HttpResponse。
5) 调用HttpResponse的getAllHeaders()、getHeaders(String name)等方法可获取服务器的响应头;调用HttpResponse的getEntity()方法可获取HttpEntity对象,该对象包装了服务器的响应内容。程序可通过该对象获取服务器的响应内容。
2.2实例:访问被保护资源
下面的Android应用需要向指定页面发送请求,但该页面并不是一个简单的页面,只有 当用户已经登录,而且登录用户的用户名是jph时才可访问该页面。如果使用HttpUrlConnection来访问该页面,那么需要处理的细节就太复杂了。下面将会借助于 HttpClient来访问被保护的页面。
访问Web应用中被保护的页面,如果使用浏览器则十分简单,用户通过系统提供的登录页面登录系统,浏览器会负责维护与服务器之间的Session,如果用户登录的用户名、密码符合要求,就可以访问被保护资源了。
为了通过HttpClient来访问被保护页面,程序同样需要使用HttpClient来登录系统,只要应用程序使用同一个HttpClient发送请求,HttpClient会自动维护与服务器之间的Session状态,也就是说程序第一次使用HttpCHent登录系统后,接下来使用HttpCHent即可访问被保护页面了。
提示:虽然此处给出的实例只是访问被保护的页面,但访问其他被保护的资源也与此类似,程序只要第一次通过HttpClient登录系统,接下来即可通过该HttpClient访问被保护资源了。
2.3程序代码:
上面的程序中①、②号粗体字代码先创建了一个HttpGet对象,接下来程序调用HttpClient的execute()方法发送GET请求;程序中③、④号粗体字代码先创建了一个HttpPost对象,接下来程序调用了HttpClient的execute()方法发送POST请求。上面的GET请求用于获取服务器上的被保护页面,POST请求用于登录系统。
运行该程序,单击“访问页面”按钮将可看到如下图所示的页面。
从上图可以看出,程序直接向指定Web应用的被保护页面secret.jsp发送请求,程序将无法访问被保护页面,于是看到下图所示的页面。单击下图所示页面中的“登录”按钮,系统将会显示如下图所示的登录对话框。
在上图所示对话框的两个输入框中分别输入“jph”、“123”,然后单击“登录”按钮,系统将会向Web站点的login.jsp页面发送POST请求,并将用户输入的用户名、密码作为请求参数。如果用户名、密码正确,即可看到登录成功的提示。
登录成功后,HttpClient将会自动维护与服务器之间的连接,并维护与服务器之间的Session状态,再次单击程序中的“访问页面”按钮,即可看到如下图所示的输出。
从上图可以看出,此时使用HttpClient发送GET请求即可正常访问被保护资源,这就是因为前面使用了HttpClient登录了系统,而且HttpClient可以维护与服务器之间的Session连接。
从上面的编程过程不难看出,使用Apache的HttpClient更加简单,而且它比HttpURLConnection提供了更多的功能。