Fresco加载gif不能显示问题解决

1.Fresco引发的血案

前面写过一篇关于Fresco分析的文章,没想到没过几天就发生了一场血案。事情是这样的,昨天另一个小哥跟我说他用Fresco加载gif不能显示了,他也不知道为什么,也没改什么东西,让我帮忙看一下。其实我们项目用Fresco已经挺久的了,然后之前那个地方也没有问题,但是不知道为什么就是不显示了。于是今天就开始了折腾。

2.解决过程

1.首先我怀疑是不是改了布局,导致控件没有显示出来。带着这样的怀疑,我设置了控件的背景为鲜明的红色,发现是显示出来的,并没有什么问题。当然这个想法可能太简单了点,但是这样的原因也是很有可能的,所以当一个控件不能显示的时候,你一定要先确定改控件是不是Visible,如果是Visible那么再去找其他原因。

2.怀疑加载图片的URI有问题,通过跟踪AbstractDraweeController中的
submitRequest()方向作为入口,不断跟踪到如下图的地方,发现URI正确,且正确调用的相应的方法。


这里写图片描述

既然url没有错,那我看看submit中的回调吧,如下图,妈蛋居然调的是失败,这我就懵逼了,这tm到底哪里出错了呢,就错了也不是咱自己的错吧,这锅不得facebook来背吗?但以我多年的职业经验来看这个事情不是那么简单,于是我继续断点反复查看。我们通过对Fresco的分析会发现,最终在处理请求时都会调用Producer类中的的produceResults来消费请求,但是对于这个Producer我也不是很懂,但是就凭他要调用的这个方法,我断点逐行分析,发现并没有什么问题。最终个条路也只能到这里结束。


这里写图片描述

3.实在没什么头绪了,于是去查看Fresco的文档http://www.fresco-cn.org/docs/getting-started.html,文档中关于加载gif的描述如下:

Uri uri;
DraweeController controller = Fresco.newDraweeControllerBuilder()
.setUri(uri)
.setAutoPlayAnimations(true)
. // 其他设置(如果有的话)
.build();
mSimpleDraweeView.setController(controller);

这代码这么简单没什么问题啊。继续看文档,发现在问题处理中说可以启动日志,既然没有办法那就启动日志来看看呗。Fresco日志默认是关闭的,启动日志方法如下,在Fresco初始化时做如下配置:

Set<RequestListener> requestListeners = new HashSet<>();
requestListeners.add(new RequestLoggingListener());
ImagePipelineConfig config = ImagePipelineConfig.newBuilder(context)
// other setters
.setRequestListeners(requestListeners)
.build();
Fresco.initialize(context, config);
FLog.setMinimumLoggingLevel(FLog.VERBOSE);

查看日志方法如下:

adb logcat -v threadtime | grep -iE 'LoggingListener|AbstractDraweeController|BufferedDiskCache'

ok打开日志后,再次加载,看到如下日志:


这里写图片描述

什么意思呢?大概就是在DecodeProducer中有一个空指针异常,调用了decodeGif在一个空引用上。于是查看DecodeProducer类找到doDecode方法,发现其中调用了mImageDecoder.decodeImage,继续查看在decodeImage中调用到了我们要找的decodeGif方法,内容如下:


这里写图片描述

根据这个方法再打断点说的就是这个mAnimatedImageFactory是空指针,那么这个AnimatedImageFactory是个什么鬼,从源码我可以知道这个类就是用来解析动图的,目前可以解析gif和webP。 继续查看调用可以看到这个factory是在ImagePipeline中创建的,如下:
这里写图片描述

就是在这里通过AnimatedFactoryProvider创建了一个AnimatedFactory,里面代码如下:


这里写图片描述

通过反射来创建的,既然出现了空指针那就是这里创建的问题咯,那我们来看看这个两个类,发现根本找不到,所以问题就是这里咯。同时发现在项目中存在两个Fresco的包0.8.0和0.11.0,于是定位到可能是依赖冲突的原因。查看了0.8.0里面创建AnimatedFactory的代码发现与0.11.0中的代码还是有点区别的,确认是依赖包冲突的问题了。
这里写图片描述

通过在Android studio Terminal输入如下命令gradlew -q app:dependencies查看当前使用的依赖版本,发现虽然配置的是使用0.8.0但是实际使用的却是0.11.0。所以解决办法1就是恢复到0.8.0版本,至于11版本的为什么没有那两个类我们稍后再来看。

+--- com.facebook.fresco:fresco:0.8.0 -> 0.11.0
|    +--- com.facebook.fresco:drawee:0.11.0
|    |    +--- com.android.support:support-v4:23.2.1 (*)
|    |    \--- com.facebook.fresco:fbcore:0.11.0
|    +--- com.facebook.fresco:fbcore:0.11.0
|    \--- com.facebook.fresco:imagepipeline:0.11.0
|         +--- com.android.support:support-v4:23.2.1 (*)
|         +--- com.facebook.fresco:fbcore:0.11.0
|         +--- com.parse.bolts:bolts-tasks:1.4.0
|         +--- com.nineoldandroids:library:2.4.0
|         \--- com.facebook.fresco:imagepipeline-base:0.11.0
|              +--- com.android.support:support-v4:23.2.1 (*)
|              +--- com.facebook.fresco:fbcore:0.11.0
|              +--- com.parse.bolts:bolts-tasks:1.4.0
|              \--- com.nineoldandroids:library:2.4.0

+--- com.facebook.react:react-native:0.29.2
|    +--- com.google.code.findbugs:jsr305:3.0.0
|    +--- org.webkit:android-jsc:r174650
|    +--- com.facebook.fresco:imagepipeline-okhttp3:0.11.0
|    |    +--- com.facebook.fresco:fbcore:0.11.0
|    |    +--- com.squareup.okhttp3:okhttp:3.0.1 -> 3.2.0
|    |    |    \--- com.squareup.okio:okio:1.6.0 -> 1.8.0
|    |    \--- com.facebook.fresco:imagepipeline:0.11.0 (*)
|    +--- com.squareup.okio:okio:1.8.0
|    +--- com.fasterxml.jackson.core:jackson-core:2.2.3
|    +--- com.squareup.okhttp3:okhttp:3.2.0 (*)
|    +--- com.facebook.fresco:fresco:0.11.0 (*)
|    +--- com.squareup.okhttp3:okhttp-ws:3.2.0
|    |    \--- com.squareup.okhttp3:okhttp:3.2.0 (*)
|    +--- com.android.support:recyclerview-v7:23.0.1 (*)
|    +--- com.squareup.okhttp3:okhttp-urlconnection:3.2.0
|    |    \--- com.squareup.okhttp3:okhttp:3.2.0 (*)
|    \--- com.android.support:appcompat-v7:23.0.1 (*)

从上面的依赖关系可以看出我们自己依赖了fresco的0.8,同时依赖了react,但是react中依赖了fresco 0.11导致我们实际依赖的是搞版本的fresco,所以如果要使用低版本的,按理说只需要把react中的fresco排除,然后我们在导入自己需要的版本就行了,但是我按照这样的思路在build.gradle进行了如下配置,结果还是使用11版本的,最终没有找到原因,你们也可以试一下,如果有发现什么问题的一定要告诉我。

compile 'com.facebook.fresco:fresco:0.8.0'
compile ('com.facebook.react:react-native:0.29.2') {
   exclude module:'com.facebook.fresco:fresco'
}

接下来我们就来看看为什么使用11版本的就少了那两个类呢?查阅各种资料,最终在一篇博客中发现了如下的内容:

一定要导入下边这个 compile 'com.facebook.fresco:animated-gif:0.12.0',否则gif图压根不动

于是瞬间懂了,原来高版本的fresco把gif相关的内容独立到了另一个依赖里面,这样如果不需要用到gif的项目就不需要导入,由于我们升级到了高版本,所以默认是没有gif相关的类的,所以就缺少了那两个类,导致gif不能显示,于是导入后,完美解决问题。

3.总结

解决问题的方法有两种:
1.退回到低版本的fresco
2.使用高版本的,需要另外导入com.facebook.fresco:animated-gif这个依赖

参考资料
gradle 管理依赖库两个问题
fesco加载gif类似问题

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,937评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,503评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,712评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,668评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,677评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,601评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,975评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,637评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,881评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,621评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,710评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,387评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,971评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,947评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,189评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,805评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,449评论 2 342

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,413评论 25 707
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,591评论 18 139
  • 图片加载是Andriod开发几乎所有应用都需要用到,做Android开发的朋友或多或少都遇到过由于图片加载过多导致...
    磨砺营IT阅读 2,708评论 -1 8
  • Fresco简单的使用—SimpleDraweeView 百学须先立志—学前须知: 在我们平时加载图片(不管是下载...
    天天大保建阅读 3,408评论 0 8
  • 在这个世界 有好多人的名字写在天空上 它们像星星一样闪耀 像云朵一样缠绵 也像雷电一样惊魂 当然,还有一些重复的名...
    甘肃子溪阅读 285评论 0 5