为什么在更新Android Studio后显示错误类型不匹配?

时间:2019-08-08 03:14:41

标签: android android-studio kotlin androidx

代码A在我的旧版本中运行良好,现在我将Android Studio更新到3.4.2,并把buildToolsVersion“ 29.0.1”更新为最新版本的androidx。

但是出现错误类型不匹配,您可以看到图像1,该如何解决?谢谢!

图片1

enter image description here

代码A

class PreferenceTool<T>(private val context: Context, private val name: String,  private val default: T) {

    private val prefs: SharedPreferences by lazy {       
        context.defaultSharedPreferences        
    }

    operator fun getValue(thisRef: Any?, property: KProperty<*>): T = findPreference(name, default)

    operator fun setValue(thisRef: Any?, property: KProperty<*>, value: T) {
        putPreference(name, value)
    }

    @Suppress("UNCHECKED_CAST")
    private fun findPreference(name: String, default: T): T = with(prefs) {
        val res: Any = when (default) {
            is Long -> getLong(name, default)
            is String -> getString(name, default)
            is Int -> getInt(name, default)
            is Boolean -> getBoolean(name, default)
            is Float -> getFloat(name, default)
            else -> throw IllegalArgumentException("This type can be saved into Preferences")
        }
        res as T
    }

    @SuppressLint("CommitPrefEdits")
    private fun putPreference(name: String, value: T) = with(prefs.edit()) {
        when (value) {
            is Long -> putLong(name, value)
            is String -> putString(name, value)
            is Int -> putInt(name, value)
            is Boolean -> putBoolean(name, value)
            is Float -> putFloat(name, value)
            else -> throw IllegalArgumentException("This type can't be saved into Preferences")
        }.apply()
    }
}

添加了内容1

还有更多,我发现代码B和代码C都可以正确编译。我不知道为什么似乎is String -> getString(name, default)会导致错误。

代码B

@Suppress("UNCHECKED_CAST")
private fun findPreference(name: String, default: T): T = with(prefs) {
    val res: Any = when (default) {
        is Long -> getLong(name, default)
        else -> throw IllegalArgumentException("This type can be saved into Preferences")
    }
    res as T
}

代码C

@Suppress("UNCHECKED_CAST")
private fun findPreference(name: String, default: T): T = with(prefs) {
    val res: Any = when (default) {
        is Long -> getLong(name, default)
        is Int -> getInt(name, default)
        is Boolean -> getBoolean(name, default)
        is Float -> getFloat(name, default)
        else -> throw IllegalArgumentException("This type can be saved into Preferences")
    }
    res as T
}

添加了内容2

似乎prefs.getString(name, default)通过图片2 返回String?。我不知道是否有Build 29.0.1的错误吗?

图片2

enter image description here

1 个答案:

答案 0 :(得分:8)

此代码无法编译,因为SharedPreferences.getString()返回String?,它不是Any的子类型。为什么以前能奏效,这是一个错误吗?

这不是错误,而是android sdk中逐渐向null安全过渡。当kotlin成为Android开发的官方语言时,they started to work on making android sdk kotlin-friendly。特别是,他们在sdk代码中添加了@Nullable@NonNull注释。但是立即这样做会破坏许多项目的编译,因此他们决定给开发人员一些时间来修复其代码:

  

通常,在Kotlin中违反可空性合同会导致编译错误。但是为了确保新注释的API与您现有的代码兼容,我们使用Kotlin编译器团队提供的内部机制将这些API标记为最近注释的API。最近带注释的API仅会导致警告,而不是Kotlin编译器的错误。您将需要使用Kotlin 1.2.60或更高版本。

     

我们的计划是让新添加的可空性注释仅产生警告,并从次年的Android SDK开始将严重性级别提高到错误。目的是为您提供足够的时间来更新代码。

正如他们所说,他们将在以下SDK中提高严重性级别,而这正是此处发生的情况。但是SharedPreferences的源代码尚未更改,那么它是怎么发生的呢?答案是@RecentlyNullable

  

(1)这里有一些注释不在支持库中,例如@RecentlyNullable和@RecentlyNonNull。这些仅在存根中用于自动将代码标记为最近用null / non-null注释的代码。我们不希望在源代码中使用这些注释。最新性是在构建时进行计算的,并代替常规的null注释注入到存根中。

尽管很久以前SharedPreferences.getString都用@Nullable进行了注释,但显然他们的自动最近性计算决定在android P中将其标记为@RecentlyNullable。为了证明这一点,让我们来看一下反编译的代码。

Android P:

@RecentlyNullable
String getString(String var1, @RecentlyNullable String var2);

Android问:

@Nullable
String getString(String var1, @Nullable String var2);