使用 Android 数据绑定时禁用 Kotlin 合成绑定

Posted

技术标签:

【中文标题】使用 Android 数据绑定时禁用 Kotlin 合成绑定【英文标题】:Disabling Kotlin synthetic bindings when using Android data binding 【发布时间】:2019-09-29 21:40:22 【问题描述】:

我正在处理this Udacity course on android/Kotlin/KTX。 “关于我”应用程序引入了数据绑定,而 Android 团队似乎 advise against using KTX binding。所以我想做前者而不是后者。

但是,事情并没有按预期进行。这些是我面临的奇怪问题:

    在我第一次通过时,我无法让 Android Studio 识别我的任何R.id.viewName ID。输入R.id. 会给我各种选项,例如R.id.action_bar,但我的活动中没有。在某些时候,这一切都解决了。 在我构建东西之前,编辑器一直抱怨数据绑定类的引用未解析(例如lateinit var binding: ActivityMainBinding)。实际上,我认为它需要不止一次构建,可能是由于我在其他地方乱七八糟的结果而得到修复(见下文)。 每个视图 ID 仍在生成为包含该 ID 的视图对象的全局变量。也就是说,如果我有一个 ID 为 doneButton 的按钮,我可以在代码中将其引用为合成的 doneButton,只需将其导入即可。我假设这意味着对象总是在运行时构建的,即使我不打算使用它们。如何使这些 KTX 样式的绑定消失?

这是我的构建 gradle:

// Top-level build file where you can add configuration options common to all sub-projects/modules.

buildscript 
    ext.kotlin_version = '1.3.31'
    repositories 
        google()
        jcenter()

    
    dependencies 
        classpath 'com.android.tools.build:gradle:3.4.0'
        classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version"
        // NOTE: Do not place your application dependencies here; they belong
        // in the individual module build.gradle files
    


allprojects 
    repositories 
        google()
        jcenter()

    


task clean(type: Delete) 
    delete rootProject.buildDir

这是我的模块 gradle:

apply plugin: 'com.android.application'

apply plugin: 'kotlin-android'

apply plugin: 'kotlin-android-extensions'

android 
    compileSdkVersion 28
    defaultConfig 
        applicationId "com.example.android.aboutme"
        minSdkVersion 19
        targetSdkVersion 28
        versionCode 1
        versionName "1.0"
        testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
    
    buildTypes 
        release 
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
        
    
    dataBinding 
        enabled = true
    


dependencies 
    implementation fileTree(dir: 'libs', include: ['*.jar'])
    implementation"org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version"
    implementation 'com.android.support:appcompat-v7:28.0.0'
    implementation 'com.android.support.constraint:constraint-layout:1.1.3'
    testImplementation 'junit:junit:4.12'
    androidTestImplementation 'com.android.support.test:runner:1.0.2'
    androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2'

在网上搜索生成数据绑定类的方法时,我在gradle.properties 中发现了a possible need to include the following line:

android.databinding.enableV2=true

我这样做后重建了,瞧!,类终于建成了。所以我正准备向 Udacity 发送有关缺失位的反馈,当我决定最好确保我了解真正解决问题的原因时。所以在成功运行程序后,我删除了上面的gradle.properties 行,清理并重建了项目。令我惊讶的是,它仍然可以构建并运行。

所以我不知道是什么解决了这个问题,也不知道如何在下一个项目中解决这些问题。

我还想阻止所有全局 KTX 绑定类被实例化,但我仍然不知道如何做到这一点。

Another SO answer 声称为生成绑定类的问题提供了两种解决方案,但给出的解决方案是针对 2016 年和 2017 年的,尽管没有实现这些解决方案,上述 gradle 文件目前正在工作(生成绑定类),所以显然有一种更新的方法。

感谢您提供的任何帮助! (哦,我还要提一下,我使用的是完全更新的 Android Studio 3.4 版。)

更新 1:我正在取得一些进展。看来,Android Studio 3.4 代码编辑器是唯一无法识别生成的数据绑定类的东西。重建时会再次识别这些类,但清洁时会丢失识别。我还不确定这种理解是否能解决我的所有问题...

更新 2:这是(至少部分)an Android Studio bug,已在 3 月修复,适用于 Canary 7。

【问题讨论】:

关于禁用 Kotlin 合成访问器,您可以尝试 ***.com/a/55643236。 “虽然 Android 团队似乎建议不要使用 KTX 绑定”——Google 既不支持也不反对它(例如,参见 this)。 谢谢。我没有找到那个 SO 问题,但答案没有改变任何东西。 如果你没有使用@Parcelize(或依赖它的东西),你可以删除apply plugin: 'kotlin-android-extensions',因为它是生成合成访问器的插件。 成功了!我想 Kotlin 扩展并没有更多的东西,对吧?我想知道为什么Android Studio 3.4默认将它们添加到项目中。 您找到了如何禁用 kotlin 合成的答案吗?我还想在我的项目中禁用它们,因为它们会导致各种运行时错误。 【参考方案1】:

为了禁用 Kotlin Synthetic 访问,请从应用级别的 build.gradle 文件中删除插件 kotlin-android-extensions。此插件生成合成导入。

【讨论】:

以上是关于使用 Android 数据绑定时禁用 Kotlin 合成绑定的主要内容,如果未能解决你的问题,请参考以下文章

使用 Kotlin for Android 进行数据绑定的问题

Android 数据绑定库 vs Kotlin Android 扩展

Kotlin-android:未解决的参考数据绑定

使用 kotlin 的 recyclerview 中的数据绑定导致问题

来自 Transformation 的数据绑定 LiveData - Android Kotlin

Android数据绑定:如何获取kotlin枚举类的字段值?