热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Android-HttpClient连接网络获取数据

大部分AndroidApp都使用HttpURLConnection和ApacheHTTPClient来发送和接收网络数据(不包括第三方开源项目),这两者都支持HTTPS,流上传和下载
    大部分Android App都使用HttpURLConnection 和 Apache HTTP Client来发送和接收网络数据(不包括第三方开源项目),这两者都支持HTTPS,流上传和下载,可配置超时,IPv6和连接池。     1)关于Apache HTTP Client:     DefaultHttpClient(android5.1\external\apache-http\src\org\apache\http\impl\client\DefaultHttpClient.java)和      AndroidHttpClient(android5.1\frameworks\base\core\java\android\net\http\AndroidHttpClient.java)都继承于HttpClient。它们有庞大且灵活的API,还比较稳定,尽管有少量的bug。然而在不破坏它们兼容性的前提下,它们庞大的API却阻碍了它们得到进一步的改进和提升。为此,Android 团队慢慢开始放弃Apache HTTP Client,在Android 6.0中,已经移除了Apache HTTP Client,在Android 官方文档中可查:http://developer.android.com/about/versions/marshmallow/android-6.0-changes.html#behavior-apache-http-client     2)关于HttpURLConnection:     HttpURLConnection的是一个通用的,适用于大多数应用的轻量级HTTP客户端。这个类出现的时候比较低调,因为其早期版本会有一些bug,但其专注的API使得它们很容易被稳步改善。在升级到Android2.2之前,HttpURLConnection有些令人沮丧的错误,特别是,在一个可读的InputStream中调用close(),可能会毒害这个链接池。为此需要禁用连接池:    
privatevoid disableConnectionReuseIfNecessary(){
// HTTP connection reuse which was buggy pre-froyo
if(Integer.parseInt(Build.VERSION.SDK) System.setProperty("http.keepAlive","false");
}
}
使用HttpURLConnection类::
        (1)使用Url.openConnection() 获得一个新的 HttpURLConnection。         (2)准备request:request主要属性是它的url。request header可以包括metadata,比如凭证(credentials),首选内容类型和session COOKIEs。         (3)选择性的上传request的主体,如果它们包括一个request body,实例对象必须设置setDoOutput(true)。被写入到stream的传输数据通过getOutputStream返回。         (4)读取response,response header通常包括metadata,如response主体内容类型和长度、修改日期和session COOKIEs。response主体可能从通过getInputStream返回的stream中读取的。如果response没有主体,该方法返回一个空的stream。         (5)断开:一旦response主体被读取到,HttpURLConnection应该调用disconnect()来断开,释放连接所持有的资源。
    综合上述,以HttpURLConnection为例来实现简单连接网络获取数据:
    1)首先在Manifest文件中增加访问网络的权限:    
 
    2)检测手机联网的状态,手机可能没有联网,连接了网络,还要区分是移动数据还是WIFI。同时在网络连接发生变化时,需要接受系统的广播:      
    android.net.conn.CONNECTIVITY_CHANGE,所以需要注册广播。这里简单的用一个类来管理网络连接的状态:  
public class NetConnectMananger {
private Context mContext;
private ConnectivityManager cm;
private NetworkInfo networkInfo;
private NetWorkChangedReceiver mNetWorkChangedReceiver;

//判断是否是WIFI连接网络
private static boolean isWifiCOnnected= false;
//判断是否是移动数据连接网络
private static boolean isMobileCOnnected= false;

public NetConnectMananger(Context context){
mCOntext= context;
cm = (ConnectivityManager) context.getSystemService(Context.CONNECTIVITY_SERVICE);
setMobileType();
//动态注册广播
mNetWorkChangedReceiver = new NetWorkChangedReceiver();
IntentFilter intentFilter = new IntentFilter();
intentFilter.addAction("android.net.conn.CONNECTIVITY_CHANGE");
context.registerReceiver(mNetWorkChangedReceiver, intentFilter);
}

public boolean isWifiNetWork(){
return isWifiConnected;
}

public boolean isMobileNetWork(){
return isMobileConnected;
}

/**
* 根据当前网络的状态来设置网络连接的TYPE
* */
private void setMobileType(){
networkInfo = cm.getActiveNetworkInfo();
if(networkInfo != null && networkInfo.getType() == ConnectivityManager.TYPE_MOBILE){
isMobileCOnnected= true;
}else if(networkInfo != null && networkInfo.getType() == ConnectivityManager.TYPE_WIFI){
isWifiCOnnected= true;
}else{
isWifiCOnnected= false;
isMobileCOnnected= false;
}
}
/**
* 取消广播的注册
* */
public void unRegisterNetWorkChangedReceiver(){
mContext.unregisterReceiver(mNetWorkChangedReceiver);
}

/**
* 接收网络连接发生改变的广播:android.net.conn.CONNECTIVITY_CHANGE
* */
public class NetWorkChangedReceiver extends BroadcastReceiver{
@Override
public void onReceive(Context context, Intent intent) {
setMobileType();
}

}  
    3)异步执行网络操作:因为在主线程执行网络操作,如果网络操作执行的时间稍长,就可能导致UI主线程线程ANR,这里以Async Task来异步执行网络操作:     
   ...
private static final String url = http://www.weather.com.cn/data/sk/101010100.html;
private NetConnectMananger ncm;
...

@Override
public void onClick(View v) {
switch (v.getId()) {
case R.id.button_start_http:
if(ncm.isMobileNetWork() || ncm.isWifiNetWork()){
new DownLoadTask().execute(url);
}else{
Toast.makeText(getApplicationContext(), "请确认网络是否连接!", Toast.LENGTH_LONG).show();
}
break;

default:
break;
}
}

@Override
protected void onDestroy() {
super.onDestroy();
//取消ncm中注册的广播。
ncm.unRegisterNetWorkChangedReceiver();
}
private class DownLoadTask extends AsyncTask{

@Override
protected String doInBackground(String... url) {

if(url != null){
try {
return downLoadData(url[0]);
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
return null;
}

@Override
protected void onPostExecute(String result) {
mTextView.setText(result);
}

public String downLoadData(String url) throws IOException{
InputStream stream = null;
int len = 500;
if(url == null){
Toast.makeText(getApplicationContext(), "无效网址!", Toast.LENGTH_LONG).show();
return null;
}
URL mUrl = new URL(url);
HttpURLConnection urlCOnnection= (HttpURLConnection) mUrl.openConnection();
try{
urlConnection.setReadTimeout(1000);
urlConnection.setConnectTimeout(20000);
urlConnection.setRequestMethod("GET");
urlConnection.setDoInput(true);

urlConnection.connect();
int respOnse= urlConnection.getResponseCode();
Log.i(TAG, "The response is: " + response);
stream = urlConnection.getInputStream();
String dataString = getStringFromStream(stream, len);
return dataString;

}finally{
if(stream != null){
stream.close();
}
if(urlConnection != null){
urlConnection.disconnect();
}
}
}
public String getStringFromStream(InputStream in, int len) throws UnsupportedEncodingException, IOException{
Reader reader = null;
reader = new InputStreamReader(in, "UTF-8");
char[] buffer = new char[len];
reader.read(buffer);
return new String(buffer);
}
}

参考:http://developer.android.com/training/basics/network-ops/connecting.html
      http://developer.android.com/reference/java/net/HttpURLConnection.html      http://developer.android.com/training/basics/network-ops/managing.html

推荐阅读
  • 本文探讨了在使用Apache Flink向Kafka发送数据过程中遇到的事务频繁失败问题,并提供了详细的解决方案,包括必要的配置调整和最佳实践。 ... [详细]
  • 本文提供了一个Android应用中用于抓取网页信息并下载图片的示例代码。通过该代码,开发者可以轻松实现从指定URL获取网页内容及其中的图片资源。 ... [详细]
  • 本文探讨了在使用 ClickOnce 部署方式时遇到的自动更新失败问题,包括本地安装与服务器安装的不同表现,并提供了详细的解决方案。 ... [详细]
  • Cadence SPB 16.5 安装指南与注意事项
    本文提供了详细的 Cadence SPB 16.5 安装步骤,包括环境配置、安装过程中的关键步骤以及常见问题的解决方案。适合初次安装或遇到问题的技术人员参考。 ... [详细]
  • 本文详细解析了Java中流的概念,特别是OutputStream和InputStream的区别,并通过实际案例介绍了如何实现Java对象的序列化。文章不仅解释了流的基本概念,还探讨了序列化的重要性和具体实现步骤。 ... [详细]
  • 本文主要解决了在编译CM10.2时出现的关于Samsung Exynos 4 HDMI HAL库中SecHdmiV4L2Utils.cpp文件的编译错误。 ... [详细]
  • 本文介绍了进程的基本概念及其在操作系统中的重要性,探讨了进程与程序的区别,以及如何通过多进程实现并发和并行。文章还详细讲解了Python中的multiprocessing模块,包括Process类的使用方法、进程间的同步与异步调用、阻塞与非阻塞操作,并通过实例演示了进程池的应用。 ... [详细]
  • 本文探讨了在使用Apache HttpClient 4.x(作为commons-httpclient 3.x的后续版本)时,如何配置默认的HttpContext,以确保每次执行请求时无需显式传递上下文。 ... [详细]
  • 本文介绍了一个基本的同步Socket程序,演示了如何实现客户端与服务器之间的简单消息传递。此外,文章还概述了Socket的基本工作流程,并计划在未来探讨同步与异步Socket的区别。 ... [详细]
  • 微服务自动化.dockercompose
    目录一、docker-compose二、docker-compose安装与配置1、修改docker.service2、下载文件3、将刚才下载的docker-compose文 ... [详细]
  • 前端技术分享——利用Canvas绘制鼠标轨迹
    作为一名前端开发者,我已经积累了Vue、React、正则表达式、算法以及小程序等方面的技能,但Canvas一直是我的盲区。因此,我在2018年为自己设定了一个新的学习目标:掌握Canvas,特别是如何使用它来创建CSS3难以实现的动态效果。 ... [详细]
  • 探讨在iOS客户端向服务器上传数据流的过程中,若服务器因权限校验失败而未接收流直接响应时,如何有效避免客户端出现超时的问题。 ... [详细]
  • 本文详细介绍了PHP中的几种超全局变量,包括$GLOBAL、$_SERVER、$_POST、$_GET等,并探讨了AJAX的工作原理及其优缺点。通过具体示例,帮助读者更好地理解和应用这些技术。 ... [详细]
  • 本文介绍如何通过Java代码调用阿里云短信服务API来实现短信验证码的发送功能,包括必要的依赖添加和关键代码示例。 ... [详细]
  • 本文介绍了如何通过安装和配置php_uploadprogress扩展来实现文件上传时的进度条显示功能。通过一个简单的示例,详细解释了从安装扩展到编写具体代码的全过程。 ... [详细]
author-avatar
手机用户2502854967
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有