前言
今天临时决定把平时积累的一些笔记整理一下,把一些值得分享的内容分享给大家,其实之前在自己的博客有发表过,不过还是在这里再次呈现一下,希望对大家有所帮助。
从笔者进入团队开始,就一直在给主管安利Fresco,在笔者的强烈要求以及对老代码深感失望之后,团队主管终于同意启用笔者推荐的Fresco,其实有经验的程序员很抗拒使用一些新的东西。当然这也可以理解,毕竟出于稳定性考虑,新的东西贸然加入,代价还是很大的,存在很大的未知性,不过加入了Fresco图片加载框架的项目不仅稳定性很好,而且各种效果都只要加一两句代码,瞬间像是发现了世外桃源,老代码中的图片加载框架在Facebook出品的牛X框架前黯然失色。
公司是全天候连着外网的,所以测试的时候并没有什么问题,另外公司还使用了一些Facebook其他的sdk,比如广告之类的,但是某个晚上,还在宿舍吃饭的笔者接到紧急电话,说图片加载一片空白(项目中有个选图的界面),其实笔者最开始的反应是网络原因,大晚上十点多回公司发现公司连不上外网了,只能访问国内的网络,笔者心里是奔溃的,因为从来没有遭遇过这种情况,最最痛苦的是连个错误都不报,哪怕来一个exception也好啊,而且Facebook出品,难道连国内的网络环境都没考虑到吗,再说之前使用的时候,连得也是国内的网络,并没有出现类似的问题。
不过笔者还是在Log中发现了一个网络访问警告,不过很遗憾没有留下截图。抓包一看,每次打开图片加载,fresco都会有graph.facebook.com 这个请求产生,笔者估计是用于数据统计之类的,不过正常的逻辑应该是即使访问失败,也应正常使用框架功能,此时boss注意到了项目中用到的其他facebook工具,诸如广告SDK之类的,而且这个加载过程是放在一个AsyncTask中来进行的。
这里粘贴下项目中使用的代码:
class MyTask extends AsyncTask<Void, Integer, Integer> {
private Context context;
UpdateTextTask(Context context) {
this.context = context;
}
@Override
protected void onPreExecute() {
}
@Override
protected Integer doInBackground(Void... params) {
LoadImages();//加载图片
return null;
}
@Override
protected void onPostExecute(Integer integer) {
}
@Override
protected void onProgressUpdate(Integer... values) {
}
}
于是考虑Asynctask阻塞的问题,我们的项目Http请求里面也是用的Asynctask,这样问题就来了。Android4.0以后Asynctask就改成(先进先出)谁先来谁先执行并且只能一个线程执行,这样就导致了所有Http请求都阻塞了。因为Asynctask执行exexute之后不会马上执行doInBackground方法,而是会去找Asynctask里面的线程池结果只有一个那就只有等待。
同样的道理,因为项目中其他的网络请求也是用Asynctask,有些访问未成功且未处理相关失败后果,请求都阻塞了,于是始终都没有执行
处理很简单,只要另外开启一个线程就完美解决了,笔者大概是这么理解的,如果有其他有更高端的方法请多多指教,欢迎指正。