记录一个小坑

1 描述

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

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

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

1
2
// 代码 1
val textView = findViewById(R.id.textView) as TextView

这是 Kotlin 的习惯写法,种写法实际上是跟下面这种写法是等价的

1
2
// 代码 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 findViewById ( id: Int ) : T!
Please specify it explicitly

报错信息

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

2 原因

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

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

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

3 解决方案:

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

3.1

在等号左边声明类型:

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

3.2

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

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

4 总结

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

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


关于作者 :
http://www.barryzhang.com
https://github.com/barryhappy
http://www.jianshu.com/users/e4607fd59d0d