objList=strList为什么报错

代码复现

先看下面一段代码

java 复制代码
List<String> strList = new ArrayList<>();
List<Object> objList = strList;  // 报错

乍一看好像没什么问题,但为什么追报错呢?

先别急,看下面一段代码

java 复制代码
List<Integer> intList = new ArrayList<>();
List<Object> objList = intList;
objList.add("hello");  // 添加某个未知类型
Integer i = intList.get(0);  // 结果一定是Integer吗

偷偷把str放到objList里了,但又从intList拿出来了?

看来你应该知道编译器为什么报错了吧,就是为了防止小白偷偷添加一些奇怪的东西

如何防止

简单啊,只要禁止add()就行了

java 复制代码
List<Integer> intList = new ArrayList<>();
List<? super Integer> objList = intList;
objList.add("hello");  // 报错
Integer i = intList.get(0); 

我们通过**? super Intege**限定了List的下界,此时objList只能存入某个Integer的父类以及父类的子类。但我们只能推断出Integer是该父类的一个子类,所以放入String会报错(String是Object的子类,但不一定是Integer某个父类的子类)。

所以编译器报错是为了防止我们乱添加?

太麻烦了?

java 复制代码
// 定义一个接口,里面只有get()
interface MyList<T> {
    T get();
}
// 实例
MyList<String> strList = () -> null;
MyList<Object> objList = strList;  // 拨错
MyList<? extends String> objList2 = strList;  // 正常

编译器报错是为了防止我们乱添加,但MyList接口里没有add()之类的添加方法,也要为了不报错而改成**? extends String**,???

Kotlin的优化

kotlin 复制代码
// 接口只有get()方法
interface MyList<out T> {
    fun get(): T
}

val strList: MyList<String> = object : MyList<String> {
    override fun get(): String {
        return "Hello"
    }
}
val objList: MyList<Any> = strList  // 正常

kotlin在语法方面做出的优化,当我们确定不会出现**偷偷把str放到objList里了,但又从intList拿出来了?**这种情况时,可以使用out关键字标记为只读不写

相关推荐
gjxDaniel2 天前
Kotlin编程语言入门与常见问题
android·开发语言·kotlin
野生技术架构师2 天前
Java 21虚拟线程 vs Kotlin协程:高并发编程模型的终极对决与选型思考
java·开发语言·kotlin
言之。2 天前
Kotlin快速入门
android·开发语言·kotlin
常利兵2 天前
Android Gradle 构建脚本现代化:Kotlin DSL (.kts) 与 Groovy DSL 深度对比与实战指南
android·开发语言·kotlin
baidu_247438612 天前
Android kotlin 定时n秒完成时回调,含暂停和继续
android·kotlin
stevenzqzq2 天前
kotlin和compose中使用by
kotlin·compose
符哥20082 天前
Android 开发中如何使用Coroutines
android·kotlin
sinat_267611913 天前
跟着官网学习协程随笔
学习·kotlin
缺一句感谢和缺一句道歉3 天前
Module was compiled with an incompatible version of Kotlin.
java·kotlin
灯火不休ᝰ3 天前
[安卓] Kotlin中的架构演进:从MVC到MVVM
android·架构·kotlin