位置: 编程技术 - 正文

Android线程交互(Handler+Thread 和 AsyncTask)(android线程间通信的几种方法)

编辑:rootadmin

推荐整理分享Android线程交互(Handler+Thread 和 AsyncTask)(android线程间通信的几种方法),希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:安卓线程间通信几种,android多线程通信,android线程机制,android线程间通信的几种方法,android中线程之间传递消息使用的是什么对象,android线程间数据传递,android线程机制,android线程与线程通信,内容如对您有帮助,希望把文章链接给更多的朋友!

为什么需要线程

  假设需要开发一个联网应用程序,需要从一个网址抓取网页内容,这里读取的网页地址是笔者在本地机器上自己建立的服务器地址。当然在读取网页内容的时候,可以使用HttpClient提供的API,但是这并不是本文的介绍重点。缺乏联网程序开发经验的程序员可能写出下面的代码。

[java] view plaincopypackage com.ophone.network; //这里为了节省篇幅,忽略了import项 public class NetworkActivity extends Activity { // 显示任务的执行状态和返回结果 private TextView message; private Button open; private EditText url; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); message = (TextView) findViewById(R.id.message); url = (EditText) findViewById(R.id.url); open = (Button) findViewById(R.id.open); open.setOnClickListener(new View.OnClickListener() { public void onClick(View arg0) { connect(); } }); } private String connect() { try { HttpClient client = new DefaultHttpClient(); // params[0]代表连接的url HttpGet get = new HttpGet(url.getText().toString()); HttpResponse response = client.execute(get); HttpEntity entity = response.getEntity(); long length = entity.getContentLength(); InputStream is = entity.getContent(); String s = null; if (is != null) { ByteArrayOutputStream baos = new ByteArrayOutputStream(); byte[] buf = new byte[]; int ch = -1; int count = 0; while ((ch = is.read(buf)) != -1) { baos.write(buf, 0, ch); count &#;= ch; // 为了在模拟器中清楚地看到进度,让线程休眠ms Thread.sleep(); } s = new String(baos.toByteArray()); } // 返回结果 return s; } catch (Exception e) { e.printStackTrace(); } return null; } }

网络连接通常是比较耗时的,尤其是在当前的GPRS这种低速率的网络情况下,这样connect()方法可能需要3-5秒,

甚至更长的时间才能返回页面的内容。如果此连接动作直接在主线程,也就是UI线程中处理,会发生什么情况呢?

为了在模拟器中更好的模拟网络读取速度慢的情况,

笔者在读取过程中让线程休眠了秒,

运行NetworkActivity,点击“连接”按钮。意外发生了,

按钮长时间没有反应,整个界面&#;乎是“死”掉了。系统随后显示出了 ANR(应用程序无响应)

错误提示,如图1所示:

在线程中联网

  为什么出现ANR?答案是联网动作阻塞在了主线程,长时间没有返回,这样OPhone弹出ANR错误。这个错误提示我们,

如果否个任务可能需要长时间的运行才能返回,则必须把这个任务放置到单独线程中运行,

避免阻塞UI线程。Java语言内置了对线程的支持,可以使用Thread类创建一个新线程,然后在run()方法中读取网页的内容,

获得页面内容后调用TextView.setText()更新界面。修改后的connect()

方法如下所示:

[java] view plaincopyprivate void connect() { new Thread() { public void run() { try { HttpClient client = new DefaultHttpClient(); // params[0]代表连接的url HttpGet get = new HttpGet(url.getText().toString()); HttpResponse response = client.execute(get); HttpEntity entity = response.getEntity(); long length = entity.getContentLength(); InputStream is = entity.getContent(); String s = null; if (is != null) { ByteArrayOutputStream baos = new ByteArrayOutputStream(); byte[] buf = new byte[]; int ch = -1; int count = 0; while ((ch = is.read(buf)) != -1) { baos.write(buf, 0, ch); count &#;= ch; } s = new String(baos.toByteArray()); message.setText(s); } } catch (Exception e) { e.printStackTrace(); } } }.start(); }

  重新运行NetworkActivity,点击“连接”按钮。程序并没有像预期的那种获得网页的内容,并显示到TextView上。查看log可以看到在connect的执行过程中抛出了异常。接下来分析问题的所在。

Android线程交互(Handler+Thread 和 AsyncTask)(android线程间通信的几种方法)

  使用Handler更新界面

  其实,connect()方法中抛出的异常是由于界面更新引起的。Connect()方法直接在新启动的线程中调用message.setText()方法是不正确的。OPhone平台只允许在主线程中调用相关View的方法来更新界面。如果返回结果在新线程中获得,那么必须借助Handler来更新界面。为此,在NetworkActivity中创建一个Handler对象,并在handleMessage()中更新UI。

[java] view plaincopy//Task在另外的线程执行,不能直接在Task中更新UI,因此创建了Handler private Handler handler = new Handler() { @Override public void handleMessage(Message msg) { String m = (String) msg.obj; message.setText(m); } };

当从connect()方法中获得网页内容后,使用如下方法更新界面。

[java] view plaincopys = new String(baos.toByteArray()); Message mg = Message.obtain(); mg.obj = s; handler.sendMessage(mg);

重新运行NetworkActivity,点击“连接”按钮,结果如图2示,正确读取了网页的内容。

 AsyncTask

  看上去修改后的connect()方法已经可用了,但是这种匿名程的方式是存在缺陷的:第一,线程的开销较大,如果每个任务都要创建一个线程,那么应用程序的效率要低很多;第二,线程无法管理,匿名线程创建并启动后就不受程序的控制了,如果有很多个请求发送,那么就会启动非常多的线程,系统将不堪重负。另外,前面已经看到,在新线程中更新UI还必须要引入handler,这让代码看上去非常臃肿。

  为了解决这一问题,OPhone在1.5版本引入了AsyncTask。AsyncTask的特点是任务在主线程之外运行,而回调方法是在主线程中执行,这就有效地避免了使用Handler带来的麻烦。阅读AsyncTask的源码可知,AsyncTask是使用java.util.concurrent 框架来管理线程以及任务的执行的,concurrent框架是一个非常成熟,高效的框架,经过了严&#;的测试。这说明AsyncTask的设计很好的解决了匿名线程存在的问题。

  AsyncTask是抽象类,子类必须实现抽象方法doInBackground(Params... p) ,在此方法中实现任务的执行工作,比如连接网络获取数据等。通常还应该实现onPostExecute(Result r)方法,因为应用程序关心的结果在此方法中返回。需要注意的是AsyncTask一定要在主线程中创建实例。AsyncTask定义了三种泛型类型 Params,Progress和Result。

Params 启动任务执行的输入参数,比如HTTP请求的URL。Progress 后台任务执行的百分比。Result 后台执行任务最终返回的结果,比如String。AsyncTask的执行分为四个步骤,与前面定义的TaskListener类&#;。每一步都对应一个回调方法,需要注意的是这些方法不应该由应用程序调用,开发者需要做的就是实现这些方法。在任务的执行过程中,这些方法被自动调用。onPreExecute() 当任务执行之前开始调用此方法,可以在这里显示进度对话框。doInBackground(Params...) 此方法在后台线程执行,完成任务的主要工作,通常需要较长的时间。在执行过程中可以调用publicProgress(Progress...)来更新任务的进度。onProgressUpdate(Progress...) 此方法在主线程执行,用于显示任务执行的进度。onPostExecute(Result) 此方法在主线程执行,任务执行的结果作为此方法的参数返回。

  PageTask扩展了AsyncTask,在doInBackground()方法中读取网页内容。PageTask的源代码如下所示:

[java] view plaincopy// 设置三种类型参数分别为String,Integer,String class PageTask extends AsyncTask<String, Integer, String> { // 可变长的输入参数,与AsyncTask.exucute()对应 @Override protected String doInBackground(String... params) { try { HttpClient client = new DefaultHttpClient(); // params[0]代表连接的url HttpGet get = new HttpGet(params[0]); HttpResponse response = client.execute(get); HttpEntity entity = response.getEntity(); long length = entity.getContentLength(); InputStream is = entity.getContent(); String s = null; if (is != null) { ByteArrayOutputStream baos = new ByteArrayOutputStream(); byte[] buf = new byte[]; int ch = -1; int count = 0; while ((ch = is.read(buf)) != -1) { baos.write(buf, 0, ch); count &#;= ch; if (length > 0) { // 如果知道响应的长度,调用publishProgress()更新进度 publishProgress((int) ((count / (float) length) * )); } // 为了在模拟器中清楚地看到进度,让线程休眠ms Thread.sleep(); } s = new String(baos.toByteArray()); } // 返回结果 return s; } catch (Exception e) { e.printStackTrace(); } return null; } @Override protected void onCancelled() { super.onCancelled(); } @Override protected void onPostExecute(String result) { // 返回HTML页面的内容 message.setText(result); } @Override protected void onPreExecute() { // 任务启动,可以在这里显示一个对话框,这里简单处理 message.setText(R.string.task_started); } @Override protected void onProgressUpdate(Integer... values) { // 更新进度 message.setText(values[0]); } }

 执行PageTask非常简单,只需要调用如下代码。重新运行NetworkActivity,不但可以抓取网页的内容,还可以实时更新读取的进度。读者尝试读取一个较大的网页,看看百分比的更新情况。

[java] view plaincopyPageTask task = new PageTask(); task.execute(url.getText().toString());

 总结

  本文介绍了OPhone联网应用开发中应该注意的两个问题:线程管理和界面更新。不但分析了问题的所在,也给出了多种解决方案。这里笔者推荐使用AsyncTask处理联网,播放大尺寸媒体文件等较为耗时的工作,不但执行效率高,也可以节省代码。

Android listView 动态加载数据,下拉加载数据,上拉加载数据 先上效果图本图为上拉到顶部实现数据加载下图为下拉到底部,实现数据加载下面展示代码,其中用到了三方开源控件pulltoRefresh,朋友们可以在网上自行

Android的AsyncTask官方API谷歌翻译版 公共抽象类AsyncTask的扩展对象的java.lang.Object↳android.os.AsyncTaskPARAMS,进展,结果类概述AsyncTask的正确实现,易于使用的用户界面线程。此类允许执行后

Android中获取JSON /***数据形式:{id:1,isNo:false,data:[{id:1,name:张三},{id:2,name:李四},{id:2,name:王五}]}*/publicstaticListMapString,StringgetJSONObject(Stringpath)throwsException{ListMapString,Stringlist=

标签: android线程间通信的几种方法

本文链接地址:https://www.jiuchutong.com/biancheng/371030.html 转载请保留说明!

上一篇:Android快速自定义控件+实战演示(三)完整键盘和组合回调!(android 自定义)

下一篇:Android listView 动态加载数据,下拉加载数据,上拉加载数据(listview安卓)

  • 征收率 税率 区别
  • 其他综合收益算营业利润吗
  • 实收资本印花税税率多少
  • 投入实收资本需要缴纳什么税
  • 核定征收企业怎么分红的
  • 哪些科目需要计提资产减值损失
  • 受托方的计税价格是什么意思
  • 餐饮业购买餐桌椅会计分录
  • 免抵退办法出口销售额和免税销售额的区别
  • 盈余公积可以随便用吗
  • 退回多交的所得税怎么做分录小规模
  • 企业购买银行理财产品如何进行账务处理
  • 外国人的纳税人识别号是护照号吗
  • 公司注销固定资产怎么处理税怎么交
  • 租金一次性付清的账务处理
  • 当无法取得对应报价时,将以即时现价报单
  • 专用发票右上角的数字表示什么
  • 简述国内采购流程
  • 以股权增资的税费怎么算
  • 私募股权投资基金管理办法最新
  • 欧拉操作系统和龙蜥哪个好用
  • 北大新闻传播学院副院长
  • 合同终止后发生的效果包括
  • 投标保证金利息规定
  • 增值税专用发票和普通发票的区别
  • 房屋出租预收租金房产税纳税义务发生时间
  • 企业所得税预缴申报
  • 事假扣款进什么科目
  • 华硕路由器登录地址
  • 应交税金 应交税费
  • php对象是什么类型的数据
  • php 自动加载
  • 哪些情况下公司不能辞退员工
  • 月末结转本年利润吗
  • 大数据找工作好找吗
  • setsid命令
  • push添加对象
  • 企业收到发票冲销怎么办
  • 2022年最新公务接待用餐标准
  • 购进免税农产品进项税额转出
  • 社会团体收取的会费是否缴纳企业所得税
  • 预缴增值税所需成本
  • 非流动资产基金 新会计制度 对应
  • css设置旋转动画
  • 织梦怎么添加相关
  • 详解PostgreSQL 语法中关键字的添加
  • 超市的库存商品是怎么记账的
  • 车辆保险费印花税计税金额含进项税吗
  • 拆迁补偿款的组成
  • 个人无需办理汇票业务
  • 工作失误扣工资的规定是什么
  • 公司注销后会计档案可以销毁吗
  • 销售费用的会计科目
  • 酒店支付清洗费属于什么会计科目
  • 三方抵账协议做什么科目
  • 为什么要用支票取钱
  • 私企公转私出纳有责任吗
  • mac查找序列号
  • win8.1怎么重新装系统
  • vmware怎么放大虚拟机
  • ubuntu如何读
  • centos重新安装命令
  • window8更新10
  • win10怎么预览
  • 微软正式推出wind...
  • xp取消开机启动项
  • 升级win10后无法修改magicbook开机画面
  • linux系统命令包含在哪个目录中
  • 禁用强制驱动程序签名有什么用
  • JAVAscript操作word
  • 游戏unity报错
  • 批处理文件自动登录远程桌面
  • lohd
  • jquery中动画效果方法
  • javascript获取复选框的值
  • 政府无偿划拨土地涉及的税费
  • 重庆车辆检测费多少钱
  • 学历认证取消申请还能再申请吗
  • 收到免税的苗木发票怎么做账
  • 苏州二套房契税交多少
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设