传统方式 ★
1 | new Thread() { |
AsnyncTask ★★
1 | new AsyncTask<String, String, String>() { |
参数类型
AsyncTask<>
的参数类型由用户设定,这里设为三个String
- 第一个String代表输入到任务的参数类型,也即是
execute()
的参数
类型 和doInBackground()
的参数
类型 - 第二个String代表处理过程中的参数类型,也就是
doInBackground()
执行过程中的产出参数类型,通过publishProgress()
发消息
传递给onProgressUpdate()
作为参数 一般用来更新进度条 - 第三个String代表任务结束的产出类型,也就是
doInBackground()
的返回值类型,和onPostExecute()
的参数类型
注意点
AsyncTask使用过程中需要注意的地方不少:
1) 由于Handler需要和主线程交互,而Handler又是内置于AsnycTask中的,所以,
AsyncTask
的创建必须在主线程
。2) AsyncTask的
doInBackground(mParams)
方法执行异步任务运行在子线程中,其他方法运行在主线程中,可以操作UI组件。3) 不要手动的去调用AsyncTask的
onPreExecute
,doInBackground
,onProgressUpdate
,onPostExecute
方法,这些都是由Android系统自动调用的一个任务AsyncTask任务只能被执行一次。4) 运行中可以随时调用
cancel(boolean)
方法取消任务,如果成功调用isCancelled()
会返回true
,并且不会执行onPostExecute()
方法了,取而代之的是调用onCancelled()
方法。而且从源码看,如果这个任务已经执行了这个时候调用cancel
是不会真正的把task
结束,而是继续执行,只不过改变的是执行之后的回调方法是onPostExecute
还是onCancelled
。
如果您的App没有明确指定屏幕方向
和configChanges
时,当用户旋转屏幕
的时候Activity就会重新启动
,而这个时候您的异步加载数据的线程可能正在请求网络。当一个新的Activity被重新创建之后,就又重新启动了一个新的任务去请求网络,这样之前的一个异步任务不经意间就泄露了,假设你还在onPostExecute
写了一些其他逻辑,这个时候就会发生意想不到异常。
一般简单的数据类型的,对付configChanges
我们很好处理,我们直接可以通过onSaveInstanceState()
和onRestoreInstanceState()
进行保存与恢复。
Android会在销毁你的Activity之前调用onSaveInstanceState()
方法,于是,你可以在此方法中存储关于应用状态的数据。
然后你可以在onCreate()
或onRestoreInstanceState()
方法中恢复。
但是,对于AsyncTask怎么办?问题产生的根源在于Activity销毁重新创建的过程中AsyncTask
和之前的Activity
失联,最终导致一些问题。那么解决问题的思路也可以朝着这个方向发展 可以用下面的两种方式
事件总线(EventBus) ★★★
使用方法参考Android中RxBus的使用
RxJava&RxAndroid ★★★★
1 | // 使用IO线程处理, 主线程响应 |
Activity销毁时取消订阅
1 |
|
总的来说现在Rx
越来越流行了 推荐新项目都用上Rx