简介Android 中的AsyncTask

时间:2022-06-19 07:53:02

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就给大家介绍到这里,希望对大家有所帮助!