首页 > 代码库 > Android开发人员应该知道的Kotlin

Android开发人员应该知道的Kotlin

本文来源于我在InfoQ中文站翻译的文章,原文地址是:http://www.infoq.com/cn/news/2016/01/kotlin-android


Android开发人员在语言限制方面面临着一个困境。

众所周知,眼下的Android开发仅仅支持Java 6(语言本身从Java 7開始进行了一些改进),因此我们每天仅仅能使用一种古老的语言来进行开发,这极大地减少了我们的生产力,同一时候也迫使我们不得不编写大量的样板与脆弱的代码。然而这种代码是难以阅读和维护的。

幸运的是,Android程序是执行在Java虚拟机之上的,因此从技术上来说,能够执行在JVM之上的一切都可用于开发Android应用。如今已经有许多能够生成JVM能够执行的字节码的语言,当中一些语言開始崭露头角并逐步流行起来,Kotlin就是当中的佼佼者。

何为Kotlin?

Kotlin是一门执行在JVM之上的语言。它由Jetbrains创建。而Jetbrains则是诸多强大的工具(如知名的Java IDE IntelliJ IDEA)背后的公司。

Kotlin是一门很easy的语言,其主要目标之中的一个就是提供强大语言的同一时候又保持简单且精简的语法。其主要特性例如以下所看到的:

  • 轻量级:这一点对于Android来说很重要。项目所须要的库应该尽可能的小。Android对于方法数量有严格的限制,Kotlin仅仅额外添加了大约6000个方法。

  • 互操作:Kotlin可与Java语言无缝通信。这意味着我们能够在Kotlin代码中使用不论什么已有的Java库;因此,即便这门语言还很年轻,但却已经能够使用成百上千的库了。除此之外,Kotlin代码还能够为Java代码所用,这意味着我们能够使用这两种语言来构建软件。

    你能够使用Kotlin开发新特性。同一时候使用Java实现代码基的其它部分。

  • 强类型:我们很少须要在代码中指定类型,由于编译器能够在绝大多数情况下判断出变量或是函数返回值的类型。这样就能获得两个优点:简洁与安全。
  • Null安全:Java最大的一个问题就是null。假设没有对变量或是參数进行null判断,那么程序当中就有可能抛出大量的NullPointerException,然而在编码时这些又是难以检測到的。Kotlin使用了显式的null,这会强制我们在必要时进行null检查。

眼下Kotlin的版本号是1.0.0 Beta 3。只是正式版很快就会公布。它全然能够用在生产当中。如今就已经有许多公司成功应用上了Kotlin。

为何说Kotlin很适合于Android?

基本上。这是由于Kotlin的全部特性都很适合于Android生态圈。

Kotlin的库很小,我们在开发过程中不会引入额外的成本。

其大小相当于support-v4库。我们在许多项目中所使用的库都比Kotlin大。

除此之外。Android Studio(官方的Android IDE)是基于IntelliJ构建的。这意味着我们的IDE对该语言提供了很棒的支持。我们能够很快就配置好项目,而且使用熟悉的IDE进行开发。

我们能够继续使用Gradle以及IDE所提供的各种执行与调试特性。这与使用Java开发应用别无二致。归功于互操作性,我们能够在Kotlin代码中使用Android SDK而不会遇到不论什么问题。实际上,部分SDK使用起来会变得更加简单,这是由于互操作性是很智能的,比方说它能够将getters与setters映射到Kotlin属性上。我们也能够以闭包的形式编写监听器。

怎样在Android开发中使用Kotlin?

过程很easy,仅仅需依照以下的步骤来就可以:

  • 从IDE plugins中下载Kotlin插件
  • 在模块中创建Kotlin类
  • 使用“Configure Kotlin in Project…”
  • 開始编码

Kotlin的一些特性

Kotlin拥有大量很打动人心的特性,这里无法一一进行介绍,只是我们来看一下当中最为重要的一些。

Null安全

如前所述,Kotlin是null安全的。

假设一个类型可能为null,那么我们就须要在类型后面加上一个?

。这样,每次在使用该类型的变量时。我们都须要进行null检查。

比方说,例如以下代码将无法编译通过:

var artist: Artist? = null?artist.print()

第2行会显示一个错误,由于没有对变量进行null检查。我们能够这样做:

if (artist != null) {? artist.print()?}

这展示了Kotlin另一个出色的特性:智能类型转换。

假设检查了变量的类型,那就无需在检查作用域中对其进行类型转换。这样。我们如今就能够在if中将artist作为Artist类型的变量了。

这对于其它检查也是适用的。另一种更简单的方式来检查null,即在调用对象的函数前使用?

。甚至还能够通过Elvis运算符?提供第二种做法:

val name = artist?.name ?

: ""

数据类

在Java中,假设想要创建数据类或是POJO类(仅仅保存了一些状态的类),我们须要创建一个拥有大量字段、getters与setters的类,或许还要提供toString与equals方法:

public class Artist { private long id; private String name; private String url; private String mbid; public long getId() { return id; } public void setId(long id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } public String getUrl() { return url; } public void setUrl(String url) { this.url = url; } public String getMbid() { return mbid; } public void setMbid(String mbid) { this.mbid = mbid; } @Override public String toString() { return "Artist{" + "id=" + id + ", name=‘" + name + ‘\‘‘ + ", url=‘" + url + ‘\‘‘ + ", mbid=‘" + mbid + ‘\‘‘ + ‘}‘; }}

在Kotlin中,上述代码能够写成以下这样:

data class Artist (?

var id: Long, var name: String, var url: String, var mbid: String)

Kotlin使用属性而非字段。

基本上,属性就是字段加上其getter与setter。

互操作

Kotlin提供了一些很棒的互操作特性。这对于Android开发帮助很大。当中之中的一个就是拥有单个方法的接口与lambda表达式之间的映射。这样,以下这个单击监听器:

view.setOnClickListener(object : View.OnClickListener { override fun onClick(v: View) { toast("Click")? }?})

能够写成这样:

view.setOnClickListener { toast("Click") }

此外。getters与setters都会自己主动映射到属性上。这并不会造成性能上的损失,由于字节码实际上仅仅是调用原来的getters与setters。

例如以下代码所看到的:

supportActionBar.title = titletextView.text = titlecontactsList.adapter = ContactsAdapter()

Lambda表达式

Lambda表达式会在极大程度上精简代码,只是重要的是借助于Lambda表达式。我们能够做到之前无法实现或是实现起来很麻烦的事情。借助于Lambda表达式,我们能够以一种更加函数式的方式来思考问题。Lambda表达式事实上就是一种指定类型,而且该类型定义了一个函数的方式。比方说。我们能够像以下这样定义一个变量:

val listener: (View) -> Boolean

该变量能够声明一个函数。它接收一个view并返回这个函数。

我们须要通过闭包的方式来定义函数的行为:

val listener = { view: View -> view is TextView }

上面这个函数会接收一个View,假设该view是TextView的实例,那么它就会返回true。由于编译器能够判断出类型。因此我们无需指定。还能够更加明白一些:

val listener: (View) -> Boolean = { view -> view is TextView }

借助于Lambda表达式,我们能够抛弃回调接口的使用。仅仅需设置希望后面会被调用的函数就可以:

fun asyncOperation(value: Int, callback: (Boolean) -> Unit) { ... callback(true)?

} asyncOperation(5) { result -> println("result: $result") }

另一种更加简洁的方式,假设函数仅仅接收一个參数。那就能够使用保留字it:

asyncOperation(5) { println("result: $it") }

Anko

Anko是Kotlin团队开发的一个库,旨在简化Android开发。其主要目标在于提供一个DSL,使用Kotlin代码来声明视图:

verticalLayout { val name = editText() button("Say Hello") { onClick { toast("Hello, ${name.text}!") } }}

它还提供了其它一些很实用的特性。比方说,导航到其它Activity:

startActivity

Android开发人员应该知道的Kotlin