传统方式 ★
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