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 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,就会发现默认的页面就会报这个错 ?…… 希望 Google 能早日改正~


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

只需五分钟,开始使用Kotlin开发Android

1:本文是一篇描述如何在Android上开始一个Kotlin的HelloWorld程序的说明文。
2:其实你如果你网络够给力的话,也许三分钟就可以了。当然网络不够给力,也可能十分钟还没整好~

好了,正文开始:


对于开发者来说,我们正处于一个美好的时代。得益于互联网的发展、工具的进步,我们现在学习一门新技术的成本和难度都比过去低了很多。
假设你之前没有使用过Kotlin,那么从头开始写一个HelloWorld的app也只需要这么几步:

首先,你要有一个Android Studio。

我正在用的是2.2.1版本,其它版本应该也大同小异。

其次,安装一个Kotlin的插件。

依次打开:Android Studio > Preferences > Plugins,然后选择『Browse repositories』,在搜索框中搜索Kotlin,结果列表中的『Kotlin』插件,就是我们要找的目标了。
点击安装,安装完成之后,重启Android Studio。
Kotlin插件

新建一个Android项目

重新打开Android Studio,新建一个Android项目吧,添加一个默认的MainActivity
——像以前一样即可。

Java to Kotlin

安装完插件的AndroidStudio现在已经拥有开发Kotlin的新能力了,那么如何体现这个能力呢?
我们先来尝试它的转换功能:Java -> Kotlin,可以把现有的java文件翻译成Kotlin文件。

打开MainActivity文件,在Code菜单下面可以看到一个新的功能:Convert Java File to Kotlin File。
Java to Kotlin

点击转换,可以看到结果:
java文件:MainActivity.java

package com.barryzhang.kotlinhello;

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }
}

转换后的Kotlin文件:MainActivity.kt

package com.barryzhang.kotlinhello

import android.support.v7.app.AppCompatActivity
import android.os.Bundle

class MainActivity : AppCompatActivity() {

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
    }
}

Kotlin的转换功能是十分实用的,对于我们重用过去的java代码、或者网上搜索到的java代码片段很有帮助。(当然,Kotlin是直接兼容java的,如果不想转换,也可以直接调用Java的方法)

配置gradle文件

MainActivity已经被转换成了Kotlin实现,但是项目目前还不可以用,还需要配置一下,让项目支持grade的编译、运行。
当然,这一步也不需要我们做太多工作——在java转换成Kotlin之后,打开MainActivity.kt文件,编译器会提示”Kotlin not configured”,点击一下Configure按钮,IDE就会自动帮我们配置好了!(所以说Kotlin的工具完善可不是吹的,毕竟Kotlin的老爹JetBrains就是专门做工具的啊)
Kotlin not configured

这个自动配置,实际上是做了这些改动。
项目的build.gradle添加:

buildscript {
    ext.kotlin_version = '1.0.4'
    dependencies { 
        classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version"
    }
}

gradle1

module里的build.gradle添加:

apply plugin: 'kotlin-android'
android { 
    sourceSets {
        main.java.srcDirs += 'src/main/kotlin'
    }
}
dependencies { 
    compile "org.jetbrains.kotlin:kotlin-stdlib:$kotlin_version"
}
repositories {
    mavenCentral()
}

gradle2

↑↑↑ 熟悉了之后自己手写也是阔以的。

Run

配置之后,等sync完成,就可以运行了~ (如果你sync失败或者耗时过长,赶紧检讨一下自己有没有科学上网?)
hello
biu~起飞,欢迎来到新世界的大门。


Also in :
http://www.barryzhang.com https://barryhappy.github.io http://www.jianshu.com/users/e4607fd59d0d/latest_articles http://blog.csdn.net/barryhappy


本文链接:http://www.barryzhang.com/?p=476