在打release
包的时候遇到了这个问题,算是比较常见,记一下解决思路
。
本文介绍了出现这个问题的原因,以及出现问题时我的建议
和我的思路
,着急的可以直接看解决方案
。
异常
代码语言:javascript复制Subscriber class xxx and its super classes have no public methods with the @Subscribe annotation
遇到异常不要慌,其实已经提示的很明显了,指定了具体的class
,且明确告诉你这个类没有@Subscribe
注解。
原因
但其实知道了问题后,可能依然会有疑惑,比如我,因为我这个类是有@Subscribe
注解的,那是为什么呢?
因为知道具体的class
,也知道@Subscribe
注解,所以很容易就能定位到是EventBus
。
随后去EventBus
官网寻找答案,果然issues
第一页就有相同的问题,作者团队是这么回复的:
Check your R8 / ProGuard rules. https://github.com/greenrobot/EventBus#r8-proguard
检查你的压缩/混淆规则。
这才反应过来,原来开启了R8
压缩导致的,minifyEnabled true
buildTypes {
release {
minifyEnabled true
...
proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
}
}
那为什么开启了R8
压缩就会导致上面的异常呢,是因为EventBus
的注解使用了反射。
反射 (Reflection) 会导致 R8 在跟踪代码时无法识别到代码的入口点。第三方库也可能用到反射,并且由于第三方库实际上是您的应用的一部分,您 (作为应用开发者) 将负责这些库以及您自己的代码中使用的反射。第三方库可能附带了它们自己的规则,但是切记,有些库不一定是为 Android 编写的,抑或是未考虑缩减问题,因此它们可能需要其他配置。
解决方案
明白了前因后果之后,解决方案就应然而生了。
方案一
关闭R8
压缩
minifyEnabled false
但是这样会使得你的应用增大不少,比如你使用了某个第三方库时,应用中只使用了其中很小一部分,但打包时所有库代码都会保留在应用中。
如果你不介意的话,这是最简单粗暴
的解决方案。
作者:https://blog.csdn.net/yechaoa
方案二
添加相应的压缩/混淆
规则,比如EventBus
:
-keepattributes *Annotation*
-keepclassmembers class * {
@org.greenrobot.eventbus.Subscribe <methods>;
}
-keep enum org.greenrobot.eventbus.ThreadMode { *; }
# And if you use AsyncExecutor:
-keepclassmembers class * extends org.greenrobot.eventbus.util.ThrowableFailureEvent {
<init>(java.lang.Throwable);
}
总结
其实问题并不复杂,关键是解决问题的过程,我想传达以下几点:
- 遇到问题先不要慌,也不要盲目搜索答案,这样会浪费时间,且可能混淆你的视线
- 先认真仔细的看一下
日志
,一般日志都会给你提示 - 结合一切可用信息,快速准确的定位问题
- 如果非要搜,建议先去
官方文档
找答案
我遇到这个异常是怎么解决的:
- 根据日志提示,我定位到问题是
EventBus
- 查看使用教程,并没有发现什么问题
- 在EventBus的
issues
中看看是否有同样的问题,果然是有的,并知道了是R8压缩
导致的 - 去
google官网
查看R8
相关介绍,并知道了原因,且提供了解决方案
举一反三,也适用其他的问题。
参考
- EventBus
- R8官方文档