Android Studio 不能识别基类中的方法

背景

  • 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 的提示看着很烦人,而且不能自动辅助弹出方法名。

can not resolve method.png

解决

可以以源码依赖的方式解决,参考 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
    }
}
can not resolve symbol.png

果然,再次印证,当遇到 A(java)=>B(kotlin)=>C(内部类) 这种情况时,AS 会出现 can not resolve xxx 的 bug。

你可能感兴趣的:(Android Studio 不能识别基类中的方法)