Android应用开发过程中必须遵循单线程模型(Single Thread Model)的原则。因为Android的UI操作并不是线程安全的,所以涉及UI的操作必须在UI线程中完成。但是并非所有的操作都能在主线程中进行,Google工程师在设计上约定,Android应用在5s内无响应的话会导致ANR(Application Not Response),这就要求开发者必须遵循两条法则:1、不能阻塞UI线程,2、确保只在UI线程中访问Android UI工具包。于是,开启子线程进行异步处理的技术方案应运而生。
以下是几种异步处理方案:
一、AsyncTask (比handler 更轻量级)
AsyncTask简介 AsyncTask的特点是任务在主线程之外运行,而回调方法是在主线程中执行,这就有效地避免了使用Handler带来的麻烦。阅读 AsyncTask的源码可知,AsyncTask是使用java.util.concurrent 框架来管理线程以及任务的执行的,concurrent框架是一个非常 成熟,高效的框架,经过了严格的测试。这说明AsyncTask的设计很好的解决了匿名线程存在的问题。 AsyncTask是抽象类,其结构图如下图所示: AsyncTask定义了三种泛型类型 Params,Progress和Result。 Params 启动任务执行的输入参数,比如HTTP请求的URL。 Progress 后台任务执行的百分比。 Result 后台执行任务最终返回的结果,比如String。 子类必须实现抽象方法doInBackground(Params… p) ,在此方法中实现任务的执行工作,比如连接网络获取数据等。通常还应 该实现onPostExecute(Result r)方法,因为应用程序关心的结果在此方法中返回。需要注意的是AsyncTask一定要在主线程中创 建实例。 AsyncTask的执行分为四个步骤,每一步都对应一个回调方法,需要注意的是这些方法不应该由应用程序调用,开发者需要做的 就是实现这些方法。在任务的执行过程中,这些方法被自动调用,运行过程,如下图所示: onPreExecute() 当任务执行之前开始调用此方法,可以在这里显示进度对话框。 doInBackground(Params…) 此方法在后台线程执行,完成任务的主要工作,通常需要较长的时间。在执行过程中可以调用 publicProgress(Progress…)来更新任务的进度。 onProgressUpdate(Progress…) 此方法在主线程执行,用于显示任务执行的进度。 onPostExecute(Result) 此方法在主线程执行,任务执行的结果作为此方法的参数返回
There are a few threading rules that must be followed for this class to work properly: The AsyncTask class must be loaded on the UI thread. This is done automatically as of JELLY_BEAN. The task instance must be created on the UI thread. execute(Params...) must be invoked on the UI thread. Do not call onPreExecute(), onPostExecute(Result), doInBackground(Params...), onProgressUpdate(Progress...) manually. The task can be executed only once (an exception will be thrown if a second execution is attempted.)
使用AsyncTask类,以下是几条必须遵守的准则:
- Task的实例必须在UI thread中创建;
- execute方法必须在UI thread中调用;
- 不要手动的调用onPreExecute(), onPostExecute(Result),doInBackground(Params...), onProgressUpdate(Progress...)这几个方法;
- 该task只能被执行一次,否则多次调用时将会出现异常;
code : (网上例子 ,直接在主线程运行这个即可 )
1. package vic.wong.main;
2. import android.os.AsyncTask;
3. import android.widget.ProgressBar;
4. import android.widget.TextView;
5.
6. /**
7. * 生成该类的对象,并调用execute方法之后
8. * 首先执行的是onProExecute方法
9. * 其次执行doInBackgroup方法
10. *
11. */
12. public class ProgressBarAsyncTask extends AsyncTask<Integer, Integer, String> {
13.
14. private TextView textView;
15. private ProgressBar progressBar;
16.
17.
18. public ProgressBarAsyncTask(TextView textView, ProgressBar progressBar) {
19. super();
20. this.textView = textView;
21. this.progressBar = progressBar;
22. }
23.
24.
25. /**
26. * 这里的Integer参数对应AsyncTask中的第一个参数
27. * 这里的String返回值对应AsyncTask的第三个参数
28. * 该方法并不运行在UI线程当中,主要用于异步操作,所有在该方法中不能对UI当中的空间进行设置和修改
29. * 但是可以调用publishProgress方法触发onProgressUpdate对UI进行操作
30. */
31. @Override
32. protected String doInBackground(Integer... params) {
33. netOperator = new NetOperator();
34. i = 0;
35. i = 10; i <= 100; i+=10) {
36. netOperator.operator();
37. publishProgress(i);
38. }
39. return i + params[0].intValue() + "";
40. }
41.
42.
43. /**
44. * 这里的String参数对应AsyncTask中的第三个参数(也就是接收doInBackground的返回值)
45. * 在doInBackground方法执行结束之后在运行,并且运行在UI线程当中 可以对UI空间进行设置
46. */
47. @Override
48. protected void onPostExecute(String result) {
49. textView.setText("异步操作执行结束" + result);
50. }
51.
52.
53. //该方法运行在UI线程当中,并且运行在UI线程当中 可以对UI空间进行设置
54. @Override
55. protected void onPreExecute() {
56. textView.setText("开始执行异步线程");
57. }
58.
59.
60. /**
61. * 这里的Intege参数对应AsyncTask中的第二个参数
62. * 在doInBackground方法当中,,每次调用publishProgress方法都会触发onProgressUpdate执行
63. * onProgressUpdate是在UI线程中执行,所有可以对UI空间进行操作
64. */
65. @Override
66. protected void onProgressUpdate(Integer... values) {
67. vlaue = values[0];
68. progressBar.setProgress(vlaue);
69. }
70.
71.
72.
73.
74.
75. }