【转】 Pro Android学习笔记(九三):AsyncTask(2):小例子
文章转载只能用于非商业性质,且不能带有虚拟货币、积分、注册等附加条件。转载须注明出处:http://blog.csdn.net/flowingflying/
小例子
通过简单的菜单,触发一个用sleep模拟的任务。Test Async One 1将调用1个AsyncTask任务,我们在Test Ansync One 2中同时执行两个task,看看运行的情况。Test Async Two中互动会更为复杂,task不仅在主线程写TextView,还开启一个ProgessDialog。
继承AsyncTask
我们先通过Test Async One 1来学习AsyncTask。首先实现继承AsyncTask。
public class MyLongTask extends AsyncTask<String,Integer,Integer>{
private IReportBack report = null;
private String tag = null;
//【步骤1】:构造函数,IReportBack定义主线程Activity处理task UI操作的回调函数,作为良好编程风格,无需将Activity对象直接传递过来。
public MyLongTask(IReportBack inr, String inTag){
report = inr;
tag = inTag;
}
@Override //【步骤2】:重写onPreExecute(),这是运行在主线程中(准确来将是运行在创建对象所在线程),进行后台线程运行先的初始化等处理。由于运行在主线程,因此可在此进行UI操作。
protected void onPreExecute() {
Log.v(tag, "onPresExecute():" + Utils.getThreadSignature());
report.reportTransient(tag, "In progress ...... "); //调用接口中的UI处理函数。
}
@Override //【步骤4】从程序员的视图,在worker线程中运行的doInBackground(),通过publishProgress()触发在主线程运行的的onProgressUpdate(),当中的机制是handler。在此,我们进行在其他线程运行时需要触发的UI处理。
protected void onProgressUpdate(Integer... values) {
Log.v(tag, "onProgressUpdate():" + Utils.getThreadSignature());
Integer i = values[0];
report.reportBack(tag, "Progress: i = " + i);//调用接口中的UI处理函数。
}
@Override //【步骤5】当后台线程运行完,通过handler将最后结果通知给线程,将在主线程触发onPostExcute(),在此进行后台处理结束触发相关的UI处理。参数result为doInbackground()的返回值。
protected void onPostExecute(Integer result) {
Log.v(tag, "onPostExecute():" + Utils.getThreadSignature());
report.reportBack(tag, "result: i = " + result);
}
@Override //【步骤3】当调用task.execute()时,将在worker线程中执行doInBackground(),由于不在主线程中执行,不能再此进行UI操作,如果希望触发相关的UI操作,通过publishProgress(),利用隐藏的handler在主线程队列加入message进行处理。主线程处理message,触发handler的handleMessage(),这些隐藏的步骤通过AsyncTask的封装,呈现为触发onProgressUpdate()。
protected Integer doInBackground(String... params) {
Log.v(tag,"doInBackground():" + Utils.getThreadSignature());
for(int i = 0; i < params.length;i ++){
Utils.sleepForSecs(2); //休眠2秒
publishProgress(i);
}
return params.length;
}
}
UI操作接口
AsyncTask封装得非常好,提高task运行前、运行中、运行后的回调函数,在此我们进行相关的UI处理。这些代码都放在对AsyncTask继承类中。处理UI就离不开Activity的View,最直接的方式是将Activity对象通过构造函数传递到AsyncTask中,然而在真正的项目开发,这样做会使一个特定的任务可以任意调用Activity的功能,两者的界面不清晰,代码可读性、可维护性,检查错误方面都不太好。因此,通常会将task涉及到的UI处理放入到接口中。
public interface IReportBack {
public void reportBack(String tag,String message);
public void reportTransient(String tag,String message);
}
使用AsyncTask
我们将在主线程中创建AsyncTask对象,下面是activity的相关代码
public class MainActivity extends Activity implements IReportBack{
private static TextView tv = null;
… …
@Override
public boolean onOptionsItemSelected(MenuItem item) {
int id = item.getItemId();
if (id == R.id.testOne1){
testMyLongTask();
}
return super.onOptionsItemSelected(item);
}
private void testMyLongTask(){
MyLongTask task = new MyLongTask(this, "TestOne"); //创建AsyncTask对象,通过这个对象,我们可以通过调用task.cancel()来停止worker线程,这时会触发onCanceled(Object)回调函数。
task.execute("Hello","my","friend"); //通过execute() 来触发后台线程执行,顺序在主线程运行onPreExecute(),然后开线程运行doInBackground()的代码,doInBackground运行结束后,触发在主线程运行的onPostExecute()。
}
@Override //接口实现
public void reportBack(String tag, String message) {
tv.append(tag + " : " + message + "\n");
Log.v(tag,message);
}
@Override //接口实现
public void reportTransient(String tag, String message) {
Toast.makeText(this, tag + " : " + message, Toast.LENGTH_SHORT).show();
reportBack(tag,message);
}
}
相关小例子源代码可在Pro Android学习:AsyncTask小例子中下载。
相关链接: 我的Android开发相关文章
【转】 Pro Android学习笔记(九三):AsyncTask(2):小例子的更多相关文章
- 【转】Pro Android学习笔记(三十):Menu(1):了解Menu
目录(?)[-] 创建Menu MenuItem的属性itemId MenuItem的属性groupId MenuItem的属性orderId MenuItem的属性可选属性 Menu触发 onOpt ...
- 【转】Pro Android学习笔记(三):了解Android资源(上)
在Android开发中,资源包括文件或者值,它们和执行应用捆绑,无需在源代码中写死,因此我们可以改变或替换他们,而无需对应用重新编译. 了解资源构成 参考阅读Android学习笔记(三八):资源res ...
- 【转】 Pro Android学习笔记(三二):Menu(3):Context菜单
目录(?)[-] 什么是Context menu 注册View带有Context menu 填Context菜单内容 Context菜单点击触发 什么是Context menu 在桌面电脑,我们都很熟 ...
- 【转】 Pro Android学习笔记(三三):Menu(4):Alternative菜单
目录(?)[-] 什么是Alternative menu替代菜单 小例子说明 Alternative menu代码 关于Category和规范代码写法 关于flags 多个匹配的itemId等参数 什 ...
- 【转】 Pro Android学习笔记(九二):AsyncTask(1):AsyncTask类
文章转载只能用于非商业性质,且不能带有虚拟货币.积分.注册等附加条件.转载须注明出处:http://blog.csdn.net/flowingflying/ 在Handler的学习系列中,学习了如何h ...
- 【转】 Pro Android学习笔记(七四):HTTP服务(8):使用后台线程AsyncTask
目录(?)[-] 5秒超时异常 AsyncTask 实现AsyncTask抽象类 对AsyncTask的调用 在哪里运行 其他重要method 文章转载只能用于非商业性质,且不能带有虚拟货币.积分.注 ...
- 【转】 Pro Android学习笔记(二九):用户界面和控制(17):include和merge
目录(?)[-] xml控件代码重用include xml控件代码重用merge 横屏和竖屏landsacpe portrait xml控件代码重用:include 如果我们定义一个控件,需要在不同的 ...
- 【转】 Pro Android学习笔记(十九):用户界面和控制(7):ListView
目录(?)[-] 点击List的item触发 添加其他控件以及获取item数据 ListView控件以垂直布局方式显示子view.系统的android.app.ListActivity已经实现了一个只 ...
- 【转】 Pro Android学习笔记(五六):配置变化
目录(?)[-] Activity的destorycreate过程 Fragment的destorycreate过程 onSaveInstanceState saveFragmentInstanceS ...
- 【转】Pro Android学习笔记(二五):用户界面和控制(13):LinearLayout和TableLayout
目录(?)[-] 布局Layout 线性布局LinearLayout 表格布局TableLayout 布局Layout Layout是容器,用于对所包含的view进行布局.layout是view的子类 ...
随机推荐
- 比较好的SQL语句
批次导数据表头 SELECT [运单号] , [运单号] AS [订单号] , [运单号] AS [订单号] , [运单号] , SUM([price] * [ProductNum]) AS [订单总 ...
- RabbitMQ 之 WorkQueues工作队列
模型图 为什么会出现 work queues? 前提:使用 simple 队列的时候 (上一篇博客)我们应用程序在是使用消息系统的时候,一般生产者 P 生产消息是毫不费力的(发送消息即可),而消费者接 ...
- selenium学习笔记(webdriver下载配置)
selenium安装后默认安装firefox可以直接使用,当然可以通过其它浏览器 博主这里整理了:chrome . IE 首先是下载地址 http://docs.seleniumhq.org/down ...
- OpenCV几种边缘检测的简例
简单记录一下OpenCV的几种边缘检测函数的用法. 边缘检测算法 以Sobel边缘检测算法为例. Sobel卷积核模板为: 偏导公式为: Gx(i,j)=[f(i+1,j−1)+2f(i+1,j)+f ...
- .html() .text() .val() 的区别
.html()用为读取和修改元素的HTML标签(包括其Html标签) .text()用来读取或修改元素的纯文本内容 (包括其后代元素) .val()用来读取或修改表单元素的value值.(只能用于表单 ...
- git远程分支回退
[本地代码回退] git reset --hard commit-id :回滚到commit-id,讲commit-id之后提交的commit都去除 git reset --hard HEAD~3:将 ...
- 你必须知道的495个C语言问题,学习体会一
C语言作为一门古老的语言,其灵活性和容易出错都让人 又爱又恨,书籍<你必须知道的495个C语言问题>,使用问答的形式,告诉读者 C语言使用的各个方面的知识,包括一些冷知识等.以下,我要摘录 ...
- ORM版,学生管理系统03
关于老师信息管理 建立多对多关系 第一种(通过外键建立) 自己写类,自己使其建立关系 缺点: 不能用Django ORM 多对多操作的语法 class Teacher(models.Model): t ...
- Luogu 2530 化工厂装箱员
Written with StackEdit. Description \(118\)号工厂是世界唯一秘密提炼锎的化工厂,由于提炼锎的难度非常高,技术不是十分完善,所以工厂生产的锎成品可能会有\(3\ ...
- Redis设计与实现 (二): 链表
Redis实现为双链表结构, 列表键的底层实现之一就是链表, 发布与订阅, 慢查询, 监视器等功能都用到了链表. Redis本身也使用链表维持多个客户端. 节点定义, 位于 adlist.h/lis ...