(转)Kotlin 跟 findViewById 的类型推导冲突问题

1 描述

从 version 26 开始,com.android.support:appcompat-v7 中的 findViewById 方法的返回值从 View 改成了 <T extends View>

对于开发者来说,喜大普奔的好处当然是以后终于可以不用在每个 finViewById 方法前面加个丑陋的类型强转了。 但是福兮祸兮,好事的背后也难免会有一些不如意的地方。

比如,如果你在用 Kotlin 的话,项目中可能会有很多类似这样的代码:

// 代码 1 
val textView = findViewById(R.id.textView) as TextView
这是 Kotlin 的习惯写法,种写法实际上是跟下面这种写法是等价的
// 代码 2
val textView : TextView = findViewById(R.id.textView) as TextView

由于 Kotlin 的类型推导特性,我们可以在声明 textView 变量的时候不必显式说明,系统会自动从后面的赋值语句中推测出它的类型是 TextView 。

但是在 version 26 之后,代码 1 的这种写法就会报错了:

Type inference failed: Not enough information to infer parameter T in
fun <T: View!> findViewById ( id: Int ) : T!
Please specify it explicitly


意思是没有足够的信息来推断 findViewById 的返回类型。

2 原因

上述错误的本质是类型推导的冲突。

如上所说,我们对 textView 的定义并没有说明其类型,它的类型是从后面的赋值语句中推导出来的。
而新版本的 findViewById ,其返回类型是<T extends View>,这是一个泛型的声明,具体类型则是根据所赋值的变量类型来确定的。

—— 等号的左右两边互相依赖,互相还都没有指明,可不就冲突报错了么!

3 解决方案:

既然是因为『两个相互依赖的类行推导都没有指明类型』,那解决方案自然就是选其中一个指明类型咯。

3.1

在等号左边声明类型:

// 代码 3
val textView : TextView = findViewById(R.id.textView)

3.2

在等号右边表明类型。
诸如这种带泛型签名的函数也是可以在调用时显式地指明类型的:

// 代码 4
val textView = findViewById<TextView>(R.id.textView)

4 总结

  1. 这只是个很简单的小问题,很好解决,但是了解其本质的过程才是更让人享受的过程~

  2. 有意思的是:As 默认支持 Kotlin 跟 findViewById 更新这两件事 —— 都是在这次的 IO 大会上宣布的。而且现在(2017.06.05)用 AS 新建一个项目并开启 Kotlin 支持,然后把 support-v7 包升级到 26,就会发现默认的页面就会报这个错

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值