很多人都会在项目中使用AsyncTask完成异步耗时的操作防止主线程被阻塞。Asynctask固然好用不过在某些情况下还是要注意。前几天项目遇到这样一个问题,老版本APP有时从广告页进到主页的过程中APP一直卡死在广告页上,当我正在查代码的时候它突然又好了并且弹出一个toast显示***已下载完成,然后跳到主页。看来应该是这个下载阻塞了广告页,不过把下载放到主线程会不会有点扯。。。后来发现在Application里面会启一个IntentService检查是否有最新版本,如果有在AsycTask里面执行一个下载最新版的task,采用的线程池是AsyncTask.SERIAL_EXECUTOR。而进入广告页时会加载一个广告图,加载完成后进入首页,加载也是采用AsyncTask,只不过是采用默认的execute方法。真相大白。。。
AsyncTask的SERIAL_EXECUTOR是SerialExecutor线程池,这种线程池的特点是使用同步机制每次只执行一个线程,其它线程需要排队等到该线程搞定后再开始执行,而AsyncTask的execute只是代理了sDefaultExecutor(SerialExecutor线程池,AsyncTask的一个静态变量)的execute
方法,所以下载任务和加载广告任务采用的同一个单线程线程池,这两个任务不是同步的你无法预判谁先谁后,所以一旦先执行下载任务APP只能等了。。。(用户可等不了,秒卸)。解决办法很简单,把加载广告task放到一个并行线程池就可以了,总之不要让它和其它任务共享一个单线程城池。
平时处理一些耗时任务可以直接用线程池就好不一定用AsycTask,而且我也考虑可以在广告页跳转处加一些逻辑让它在一定时间强行跳转避免因为某某原因而卡在那里。
网友评论