背景
- Android Studio:
Android Studio Dolphin | 2021.3.1 Patch 1
- Gradle:
distributionUrl=https\://services.gradle.org/distributions/gradle-7.3.3-bin.zip
- AGP:
plugins {
id 'com.android.application' version '7.2.2' apply false
id 'com.android.library' version '7.2.2' apply false
id 'org.jetbrains.kotlin.android' version '1.6.10' apply false
}
问题
使用了 implementation 'com.github.CymChad:BaseRecyclerViewAdapterHelper:3.0.8'
,其依赖了 implementation 'androidx.recyclerview:recyclerview:1.2.1'
,在新建的 MyAdapter 类中调用 RecyclerView.ViewHolder#getAdapterPosition() 方法时,Android Studio 提示 can not resolve method
。
注意这里是不影响正常编译和运行的,只是 AS 的提示看着很烦人,而且不能自动辅助弹出方法名。
解决
可以以源码依赖的方式解决,参考 Android 依赖项目外部文件夹中的工程 - (jianshu.com)
实验
为了弄清楚什么情况下会出现这种情况,做了如下实验:
新建一个 Test 工程,除 app 模块外,再建一个 lib-aar 模块和 lib-rv 模块,其中 lib-rv 模块模拟 RecyclerView,lib-aar 模块模拟 BaseRecyclerViewAdapterHelper。
| - app
| - MyHolder1JavaExtendsJava.java
| - MyHolder1JavaExtendsKotlin.java
| - MyHolder1KotlinExtendsJava.kt
| - MyHolder1KotlinExtendsKotlin.kt
| - MyHolder2JavaExtendsJava.java
| - MyHolder2JavaExtendsKotlin.java
| - MyHolder2KotlinExtendsJava.kt
| - MyHolder2KotlinExtendsKotlin.kt
| - lib-aar
| - BaseHolder1Java.java
| - BaseHolder1Kotlin.kt
| - BaseHolder2Java.java
| - BaseHolder2Kotlin.kt
| - lib-rv
| - MyRecyclerView.java
| - ViewHolder2.java
| - maven
- MyRecyclerView.java
public class MyRecyclerView {
public abstract static class ViewHolder1 {
public int getAdapterPosition() {
return 0;
}
}
}
- ViewHolder2.java
public class ViewHolder2 {
public int getAdapterPosition() {
return 0;
}
}
- BaseHolder1Java.java
public class BaseHolder1Java extends MyRecyclerView.ViewHolder1 {
}
- BaseHolder1Kotlin.kt
open class BaseHolder1Kotlin : ViewHolder1() {
}
- BaseHolder2Java.java
public class BaseHolder2Java extends ViewHolder2 {
}
- BaseHolder2Kotlin.kt
open class BaseHolder2Kotlin : ViewHolder2() {
}
- MyHolder1JavaExtendsJava.java
public class MyHolder1JavaExtendsJava extends BaseHolder1Java {
public void test() {
int pos = this.getAdapterPosition();
}
}
- MyHolder1JavaExtendsKotlin.java 该文件提示
can not resolve method
public class MyHolder1JavaExtendsKotlin extends BaseHolder1Kotlin {
public void test() {
int pos = this.getAdapterPosition();
}
}
- MyHolder1KotlinExtendsJava.kt
class MyHolder1KotlinExtendsJava : BaseHolder1Java() {
fun test() {
val pos = this.getAdapterPosition()
}
}
- MyHolder1KotlinExtendsKotlin.kt
class MyHolder1KotlinExtendsKotlin : BaseHolder1Kotlin() {
fun test() {
val pos = this.getAdapterPosition()
}
}
- MyHolder2JavaExtendsJava.java
public class MyHolder2JavaExtendsJava extends BaseHolder2Java {
public void test() {
int pos = this.getAdapterPosition();
}
}
- MyHolder2JavaExtendsKotlin.java 该文正常
public class MyHolder2JavaExtendsKotlin extends BaseHolder2Kotlin {
public void test() {
int pos = this.getAdapterPosition();
}
}
- MyHolder2KotlinExtendsJava.kt
class MyHolder2KotlinExtendsJava : BaseHolder2Java() {
fun test() {
val pos = this.getAdapterPosition()
}
}
- MyHolder2KotlinExtendsKotlin.kt
class MyHolder2KotlinExtendsKotlin : BaseHolder2Kotlin() {
fun test() {
val pos = this.getAdapterPosition()
}
}
结论
通过上面的对比可以发现,当我们有一个类 A,其继承自一个 aar 中(说明不是源码引用)的类 B,而类 B 又继承自一个类 C,那么当:
- 类 A 以 java 代码编写
- 类 B 以 kotlin 代码编写且位于 aar 中
- 类 C 是一个内部类
满足以上三个条件时,在 AS 中无法会出现在类 A 源码中无法解析类 C 中的方法的情况,即类 A 中调用类 C 的方法,提示 can not resolve method
。
至此,算是知道了什么情况下会出现该问题,也知道了如何避免该问题的方法,要么以源码形式依赖,要么避免出现 A(java)=>B(kotlin)=>C(内部类)
的情况即可。
至于为什么会出现此问题,还需要 Google 给出原因。。。
补充
上面推断出是 A(java)=>B(kotlin)=>C(内部类)
这种情况导致的,于是想到再找个普通的类进行测试,想一想哪些类中有内部类呢,首先想到的就是 FrameLayout$LayoutParams
:
- BaseLayoutParams.kt
open class BaseLayoutParams : FrameLayout.LayoutParams(0, 0) {
}
- MyLayoutParamsJava.java 不正常
public class MyLayoutParamsJava extends BaseLayoutParams {
public void test() {
int g = this.gravity;
}
}
MyLayoutParamsKotlin.kt 正常
class MyLayoutParamsKotlin : BaseLayoutParams() {
fun test() {
val g = this.gravity
}
}
果然,再次印证,当遇到 A(java)=>B(kotlin)=>C(内部类)
这种情况时,AS 会出现 can not resolve xxx
的 bug。