asynctask是一个很常用的api,尤其异步处理数据并将数据应用到视图的操作场合。其实asynctask并不是那么好,甚至有些糟糕。本文我会讲asynctask会引起哪些问题,如何修复这些问题,并且关于asynctask的一些替代方案。
asynctask
从android api 3(1.5 cupcake)开始,asynctask被引入用来帮助开发者更简单地管理线程。实际上在android 1.0和1.1也是有类似的实现,那就是usertask。usertask和asynctask有着相同的api及实现,但是由于由于1.0和1.1的设备份额微乎其微,这里的概念就不会涉及到usertask。
生命周期
关于asynctask存在一个这样广泛的误解,很多人认为一个在activity中的asynctask会随着activity的销毁而销毁。然后事实并非如此。asynctask会一直执行doinbackground()方法直到方法执行结束。一旦上述方法结束,会依据情况进行不同的操作。
如果cancel(boolean)调用了,则执行oncancelled(result)方法
如果cancel(boolean)没有调用,则执行onpostexecute(result)方法
asynctask的cancel方法需要一个布尔值的参数,参数名为mayinterruptifrunning,意思是如果正在执行是否可以打断, 如果这个值设置为true,表示这个任务可以被打断,否则,正在执行的程序会继续执行直到完成。如果在doinbackground()方法中有一个循环操作,我们应该在循环中使用iscancelled()来判断,如果返回为true,我们应该避免执行后续无用的循环操作。
总之,我们使用asynctask需要确保asynctask正确地取消。
asynctask和handler对比
1 ) asynctask实现的原理,和适用的优缺点
asynctask,是android提供的轻量级的异步类,可以直接继承asynctask,在类中实现异步操作,并提供接口反馈当前异步执行的程度(可以通过接口实现ui进度更新),最后反馈执行的结果给ui主线程.
使用的优点:
简单,快捷
过程可控
使用的缺点:
在使用多个异步操作和并需要进行ui变更时,就变得复杂起来.
2 )handler异步实现的原理和适用的优缺点
在handler 异步实现时,涉及到 handler, looper, message,thread四个对象,实现异步的流程是主线程启动thread(子线程)àthread(子线程)运行并生成message-àlooper获取message并传递给handleràhandler逐个获取looper中的message,并进行ui变更。
使用的优点:
结构清晰,功能定义明确
对于多个后台任务时,简单,清晰
使用的缺点:
在单个后台异步处理时,显得代码过多,结构过于复杂(相对性)
asynctask介绍
android的asynctask比handler更轻量级一些,适用于简单的异步处理。
首先明确android之所以有handler和asynctask,都是为了不阻塞主线程(ui线程),且ui的更新只能在主线程中完成,因此异步处理是不可避免的。
android为了降低这个开发难度,提供了asynctask。asynctask就是一个封装过的后台任务类,顾名思义就是异步任务。
asynctask直接继承于object类,位置为android.os.asynctask。要使用asynctask工作我们要提供三个泛型参数,并重载几个方法(至少重载一个)。
asynctask定义了三种泛型类型 params,progress和result。
•params 启动任务执行的输入参数,比如http请求的url。
•progress 后台任务执行的百分比。
•result 后台执行任务最终返回的结果,比如string。
使用过asynctask 的同学都知道一个异步加载数据最少要重写以下这两个方法:
•doinbackground(params…) 后台执行,比较耗时的操作都可以放在这里。注意这里不能直接操作ui。此方法在后台线程执行,完成任务的主要工作,通常需要较长的时间。在执行过程中可以调用publicprogress(progress…)来更新任务的进度。
•onpostexecute(result) 相当于handler 处理ui的方式,在这里面可以使用在doinbackground 得到的结果处理操作ui。 此方法在主线程执行,任务执行的结果作为此方法的参数返回
有必要的话你还得重写以下这三个方法,但不是必须的:
•onprogressupdate(progress…) 可以使用进度条增加用户体验度。 此方法在主线程执行,用于显示任务执行的进度。
•onpreexecute() 这里是最终用户调用excute时的接口,当任务执行之前开始调用此方法,可以在这里显示进度对话框。
•oncancelled() 用户调用取消时,要做的操作
使用asynctask类,以下是几条必须遵守的准则:
•task的实例必须在ui thread中创建;
•execute方法必须在ui thread中调用;
•不要手动的调用onpreexecute(), onpostexecute(result),doinbackground(params...), onprogressupdate(progress...)这几个方法;
•该task只能被执行一次,否则多次调用时将会出现异常;
一个超简单的理解 asynctask 的例子:
main.xml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
|
<?xml version= "." encoding= "utf-" ?>
<linearlayout xmlns:android= "http://schemas.android.com/apk/res/android"
android:orientation= "vertical"
android:layout_width= "fill_parent"
android:layout_height= "fill_parent"
>
<textview
android:id= "@+id/textview"
android:layout_width= "fill_parent"
android:layout_height= "wrap_content"
/>
<progressbar
android:id= "@+id/progressbar"
android:layout_width= "fill_parent"
android:layout_height= "wrap_content"
style= "?android:attr/progressbarstylehorizontal"
/>
<button
android:id= "@+id/button"
android:layout_width= "fill_parent"
android:layout_height= "wrap_content"
android:text= "更新progressbar"
/>
</linearlayout>
|
mainactivity.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
|
package vic.wong.main;
import android.app.activity;
import android.os.bundle;
import android.view.view;
import android.view.view.onclicklistener;
import android.widget.button;
import android.widget.progressbar;
import android.widget.textview;
public class mainactivity extends activity {
private button button;
private progressbar progressbar;
private textview textview;
@override
public void oncreate(bundle savedinstancestate) {
super .oncreate(savedinstancestate);
setcontentview(r.layout.main);
button = (button)findviewbyid(r.id.button);
progressbar = (progressbar)findviewbyid(r.id.progressbar);
textview = (textview)findviewbyid(r.id.textview);
button.setonclicklistener( new onclicklistener() {
@override
public void onclick(view v) {
progressbarasynctask asynctask = new progressbarasynctask(textview, progressbar);
asynctask.execute();
}
});
}
}
|
netoperator.java
1
2
3
4
5
6
7
8
9
10
11
12
13
|
package vic.wong.main;
//模拟网络环境
public class netoperator {
public void operator(){
try {
//休眠秒
thread.sleep();
} catch (interruptedexception e) {
// todo auto-generated catch block
e.printstacktrace();
}
}
}
|
progressbarasynctask .java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
|
package vic.wong.main;
import android.os.asynctask;
import android.widget.progressbar;
import android.widget.textview;
/**
* 生成该类的对象,并调用execute方法之后
* 首先执行的是onproexecute方法
* 其次执行doinbackgroup方法
*
*/
public class progressbarasynctask extends asynctask<integer, integer, string> {
private textview textview;
private progressbar progressbar;
public progressbarasynctask(textview textview, progressbar progressbar) {
super ();
this .textview = textview;
this .progressbar = progressbar;
}
/**
* 这里的integer参数对应asynctask中的第一个参数
* 这里的string返回值对应asynctask的第三个参数
* 该方法并不运行在ui线程当中,主要用于异步操作,所有在该方法中不能对ui当中的空间进行设置和修改
* 但是可以调用publishprogress方法触发onprogressupdate对ui进行操作
*/
@override
protected string doinbackground(integer... params) {
netoperator netoperator = new netoperator();
int i = ;
for (i = ; i <= ; i+=) {
netoperator.operator();
publishprogress(i);
}
return i + params[].intvalue() + "" ;
}
/**
* 这里的string参数对应asynctask中的第三个参数(也就是接收doinbackground的返回值)
* 在doinbackground方法执行结束之后在运行,并且运行在ui线程当中 可以对ui空间进行设置
*/
@override
protected void onpostexecute(string result) {
textview.settext( "异步操作执行结束" + result);
}
//该方法运行在ui线程当中,并且运行在ui线程当中 可以对ui空间进行设置
@override
protected void onpreexecute() {
textview.settext( "开始执行异步线程" );
}
/**
* 这里的intege参数对应asynctask中的第二个参数
* 在doinbackground方法当中,,每次调用publishprogress方法都会触发onprogressupdate执行
* onprogressupdate是在ui线程中执行,所有可以对ui空间进行操作
*/
@override
protected void onprogressupdate(integer... values) {
int vlaue = values[];
progressbar.setprogress(vlaue);
}
}
|
关于android 中的asynctask就给大家介绍到这里,希望对大家有所帮助!