在打​​release​​​包的时候遇到了这个问题,算是比较常见,记一下​​解决思路​​。

本文介绍了出现这个问题的原因,以及出现问题时​​我的建议​​​和​​我的思路​​​,着急的可以直接看​​解决方案​​。

异常

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

但是这样会使得你的应用增大不少,比如你使用了某个第三方库时,应用中只使用了其中很小一部分,但打包时所有库代码都会保留在应用中。

如果你不介意的话,这是最​​简单粗暴​​的解决方案。

作者: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);
}

总结

其实问题并不复杂,关键是解决问题的过程,我想传达以下几点:

  1. 遇到问题先不要慌,也不要盲目搜索答案,这样会浪费时间,且可能混淆你的视线
  2. 先认真仔细的看一下​​日志​​,一般日志都会给你提示
  3. 结合一切可用信息,快速准确的定位问题
  4. 如果非要搜,建议先去​​官方文档​​找答案

我遇到这个异常是怎么解决的:

  1. 根据日志提示,我定位到问题是​​EventBus​
  2. 查看使用教程,并没有发现什么问题
  3. 在EventBus的​​issues​​​中看看是否有同样的问题,果然是有的,并知道了是​​R8压缩​​导致的
  4. 去​​google官网​​​查看​​R8​​相关介绍,并知道了原因,且提供了解决方案

举一反三,也适用其他的问题。

如果对你有用或启发的话,点个赞呗 ^ _ ^

参考