项目开发快到尾声,突然发现之前一个模块莫名其妙的奔溃了,我的内心也是奔溃的。以前一直都是好好的,也没去动过它,为啥会出现这样的问题呢? 下面我会根据自己的理解来看待问题
android是怎么根据id查找到控件的
首先,你在调用 findViewById 之前,你必然是在 activity 中设置了 setContentView, 或者在 Fragment 中重载了 onCreatedView 方法,对于 findViewById, 他只能使用在 view或者 activity 下,对于 view, 你进行遍历的根节点就是对应的 view, 对于 activity, 你对应的根节点就是你使用 setContentView 初始化的布局
当你调用 findViewById 是, android 先对比本身是否具有该 id,是则返回自己,不是则判断自己是否为 ViewGroup, 如果是再对子视图进行遍历,否则返回 null, 遍历时,按照从上到下的顺序一一遍历,只要找到一个节点的 id 为搜索的 id, 则返回这个节点代表的 view, 比如说你的 layout 中有两个相同 id 的 view, 那么返回的必然是最前的那一个
为什么会出现这样的情况
首先我们看看LOG日子
java.lang.NoSuchFieldError: No static field tabTexts of type I in class Lcom/xxx/R$id; or its superclasses (declaration of 'com.xxx.R$id' appears in /data/data/com.sss/files/instant-run/dex/slice-slice_1-classes.dex)
这是关键问题,这里告诉我们没有找到tabTexts 的字段(ID),奇怪了,我也没有碰过,为啥会突然出现这样的问题。我们可以从这方面入手查找问题。
解决问题
上面说到了,调用findViewById 时会对相应的layout进行遍历查找,如果没有则返回null。同样的NoSuchFieldError是Java反射中的一个异常,其表示无法通过反射找到需要的字段。进行到这里,我们该考虑了,是不是加载的时候,不是加载了相应的layout。导致找不到tabTexts 的ID控件。于是全局搜索一下layout的名字,终于在这里发现了弊端。我们来看看搜索结果:
这里我们可以发现,我滴天。怎么会有两个不同的ID,问题果然出现在这里。项目在加载layout的时候,默认加载了0x7f0d01c6的ID,导致找不到tabTexts 。这下好解决了。BUG哪里跑,看我还不消灭你。
解决办法:
把custom_tab(就是报错的layout)换一个名字。避免android加载错误的layout.