谈Kotlin

简洁的语法

Kotlin给我的感觉就是它就是针对Java里所有的痛点来做改变的,比如语法繁琐,烦人的空指针,缺少函数式编程支持(Java8虽然支持Lambda表达式但还远远不够)等等。不像它的孪生兄弟Scala那样奔放,Kotlin给我的感受是聪明又克制,对Java程序员来说半天就足够上手开干了,不像Scala门槛那么高。

Java语言是我所接触过语言中语法最繁琐的(有更繁琐的请告知),这种繁琐在main函数就可以体现,而且Java可能也是唯一一个几乎没法不用IDE的语言。其他语言脱离了IDE的语法警告和代码生成模版也勉强能写一写,Java代码你没有试试看?

但是Java的语法之严谨和繁琐恰恰是Java如此流行的原因。没写过Java的人可能没法理解这句话。人们对于Java严苛的语法还不够满意,还要再加上check-style,findbugs以及各种XXX公司Java代码规范等各种条条框框,最终得到的结果是所有Java程序员写出来的代码都能互相看懂。即使是刚毕业不久的Java程序员,只要熟悉Java面向对象语法,几乎不存在看不懂其他Java代码的情况(业务看不懂除外)。这样保证了Java程序员的代码下限非常之高,能够保证大型项目的成功。

好像扯Java扯的有点多了。接触Kotlin之后会发现,kotlin在尽可能保持和Java一样的语法严苛性的情况下,大幅度地精简了Java代码,写起来非常之爽。

//快速创建一个Bean类
data class Person(val id: Long, var name: String, var age: Int)

fun main() {
    // 创建实例
    val zhangsan = Person(0L, "zhangsan", 23)
    // set/get方法
    zhangsan.age = 24
    val name = zhangsan.name
    
    //copy方法
    val lisi = zhangsan.copy(id = 1, name = "lisi")
    //默认生成的tostring和equals/hashcode方法,可重写
    zhangsan.toString()
    zhangsan.hashCode()
    var isEqual = zhangsan.equals(lisi)
}

简洁的语法无处不在,比如上面的Bean类如果用java来写至少要多好几倍的代码。单例模式只需要把class换成object即可。这些虽然用Java的IDE的各种模版生成工具一样可以秒生成,但是还是kotlin写起来更舒服一些。

我想强调的是,Kotlin并不是无脑地利用语法糖和新关键字来精简语法,我们可以很清晰地感受到Kotlin在设计过程中尽可能地保证它尽可能的语法严苛性。

空指针

Java的空指针异常被称之为“Billion Dollar Mistake”。在上家公司的时候为了避免空指针异常,代码规范要求几乎所有的对象都要进行判空操作,那种嵌套很深的json对象,判空起来真的是又臭又长。

val name: String?
//这里要吐槽下为啥kotlin没有三目运算符?
name = if(Random.nextBoolean()) "Ann" else null
//加上?则只在name不为null的时候会调用,得到的len1类型也是有问号的,即Int?
val len1 = name?.length
//这种写法是如果为空给一个默认值,得到的len2类型为Int,不带问号
val len2 = name?.length?:0
//加上双感叹号其实就是java什么都不加的写法了,如果为null会报空指针异常
val len3 = name!!.length

上面的代码解释了三种对可能为空的对象的处理。其实我把Kotlin对空指针的处理归类为给所有对象加了个范型——符号?,对于带?的对象则表明它可能为空,你不处理直接使用的话不让编译,也就是把空指针尽量扼杀在编译期间。这个想法真的是简单又巧妙。很多人一听Kotlin就说没有空指针了,但是没用过的话并不知道怎么实现的。现在再有面试官问你,知道怎么回答了吧?

函数式编程

fun AppCompatActivity.setupActionBar(@IdRes toolbarId: Int, action: ActionBar.() -> Unit) {
    val toolbar = findViewById<Toolbar>(toolbarId)
    setSupportActionBar(toolbar)
    supportActionBar?.run {
        //执行参数中的函数,这样用户在调用该方法的时候更加灵活
        action()
    }
    toolbar.setNavigationOnClickListener {
        Log.d("AppCompatActivity", "finish")
        finish()
    }
}

//------------------------分割线-----------------------------
//类似的单方法接口现在只需要写一个闭包就行了
binding.aliPayIcon.setOnClickListener {
    Log.d("example", it.contentDescription.toString())
}

//扩展函数let,只有在对象不为空的时候会调用,相当于做了判空
binding.let {
    it.setLifecycleOwner(this@WithdrawActivity)
    it.viewModel = vm
}

//扩展函数apply, 在闭包内可直接调用对象的方法属性,有个好处就是可以直接操作对象不需要先生成变量
vm.accountName.apply {
    this.value = "aaaa"
    Log.d("example", this.value?.toString() + ":" + this.hashCode())
}

//还有其他基础扩展函数run, with, also等等,可以看看这篇博客的介绍:https://www.jianshu.com/p/28ce69d58fea

函数式编程对很多Java程序员来说是很陌生的。尽管我不想黑,但是Java真的强到让很多Java程序员只会Java一门语言,并且基本上Java8以上的版本也没接触过。而除了Java(Java8以下)我还真不知道哪门语言不支持函数式编程。

几乎所有语言都支持函数式编程,因此会灵活使用函数式编程真的挺重要。Java不支持是因为函数式编程相对要难一点,不支持函数式编程反而让Java代码的下限更高。除了难度大一点,函数式编程最令人不爽的就是阅读性差(可能比较依赖写代码的人的水平),我看Java框架源码基本都很顺畅,但是看JavaScript框架简直懵逼。我现在的水平看Kotlin函数式代码也费劲,但是kotlin毕竟是强类型语言,函数的参数和返回值类型固定的话,认真读还是不难读懂的。

扯远了,函数式编程的好处就是代码灵活度加倍提高,可以写出各种秀操作的代码,显然这也是一把双刃剑。不过随着函数式编程越来越流行,大家的水平越来越高,写出来的函数式代码势必越来越好。

可以说Java程序员学习Kotlin唯一的难点就是函数式编程了,当然如果你本身就有函数式编程经验这点也不是事。如果没有的话就要好好学习参悟下函数式编程了,这是一个相对漫长的过程。在熟练掌握之前,不使用函数式编程也一样可以写Kotlin。

结语

总的来说我对Kotlin的评价总体是正面的。不过有些人的脑回路不正常,我还是想说一下:说Kotlin好,有优点,马上又有人惊呼Java要被取代啦!过两天又有人大喊Kotlin怎么怎么不好马上要凉。Java的好处和适用范围相信你也明白。而现在还在大量使用的语言都有它各自的优点,这个我就不唠了,毕竟水平不够。我是真恶心某些蹭热度、发软文各自骗的博主和公众号。

不要听信我和谷歌说它好,也不要听信那些碰瓷博主说它不好,有那时间坐而论道,为啥不抽出半小时来体验一下Kotlin呢?


作者:业松
链接:https://www.jianshu.com/p/5961116ddba6
来源:简书
 

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值