开启Kotlin编程之旅&Java程序员的思维进化

这周在部门进行了一场Kotlin分享,于是把分享内容整理出这篇文章

Kotlin编程语言简介

  • 由Intelij IDEA生产商JetBrains开源
  • 2011年开始,2016年初发布1.0正式版,目前最新1.2.31
  • 基于JVM平台,JS平台和Native的编程语言
  • 静态的,支持函数式编程范式
  • 与Java语言极高的兼容和互操作

Kotlin的生态环境

  • GitHub star 2万多
  • Kotlin在2018 TIOBE 3月份语言排行榜38名
  • 国外Stackoverflow 发布的2018 Developer Survey Results报告中,75.1%对Kotlin感兴趣
  • Spring 5.0版本将支持Kotlin
  • Kotlin不足:编译器存在优化空间,大范围使用目前只在Android开发领域

Kotlin的语言特性&Kotlin与Java的差异

空指针安全

引用/对象为空时调用报NullPointException异常

Java世界里

Java对空指针的处理的方式有以下:

  1. if判断,过滤null,缺陷:代码冗余

  2. 做一层包装,比如Java世界里,Double,Int 装箱;Java8里Optional包装; 缺陷:代码冗余,额外的包装接口影响运行时性能,即使在代码中到处都使用了Optional,仍然需要处理JDK、Android框架,以及其他第三方库中的方法返回null值。

  3. 使用注解(@nullable,@NotNull) + 插件代码检测;缺陷:这些工具不是标准Java编译过程的一部分,很难保证她们自始至终都被应用,而且很难使用注解标记覆盖所有可能发生错误的地方

Kotlin要怎么做的?

Kotlin世界里

以字符串对象为例

在Java里:

String = String + null

在Kotlin里

String = String  
String?= String + null

因此,String和String?是两种类型。Kotlin中的String?相当于Java里的String

对类型,Kotlin让我们有了新的认识:

  1. Kotlin中,所有常见类型默认都是非空的。
  2. 什么是类型?就是对数据的分类,分类的类目里有一类是null

隆重登场, Kotlin的处理

程序执行顺序上出了偏差或是其他原因,对象引用没有就建立起来,就会出现空指针,空指针在逻辑上就存在了

  1. ?.调用 把一次null检查和一次方法调用合并成一个操作。
  2. ?: Evlis运算符:问号前面的对象是null么?如果是则返回冒号后面的值,如果不是则返回问号前的值
  3. 拓展函数
data class Person(val name: String, val age: Int){

    fun walk(){
        println("$name is walking")
    }
}

val p = Person("Kotlin",6)
val p2 = null
p2?.walk()

val p3 = p2?:p
println("${p3.name}")

简洁、高效性

  • 好吃的语法糖,少了Java冗余啰嗦;比如类型推断与自动强转,引入数据类data
  • 语言层级提供了大量的非常方便的实现;比如Kotlin的标准库封装了大量对集合操作的快捷方法
  • 命名参数,默认参数
  • 拓展函数
  • 高阶函数,局部函数实现闭包

kotlin提供了一些特性保证Kotlin简洁高效,比如:

拓展函数

StringUtil.captitalize(s)
s.captitalize

意义:一方面让代码组织的更简洁,另一方面暗合了Java6大设计原则的开闭原则。对原来的类的定义不做修改,而是通过拓展特性来完成(java实现上是通过工具类组合的方式来做的)

运算符重载

set.add(2)
set += 1

中缀调用

1.to("one")
1 to one

get方法约定

当作成员变量般调用

map.get("key")
map["key"]

invoke约定/对()操作符的重载

有了这个约定,在Kotlin的世界里,一切对象都可以认为是函数了。比如lambda表达式,可以这样使用labmda()就是因为
Kotlin约定了,除非是内联,lambda表达式都会被编译成实现了函数式接口(Function1)的类,这些接口定义了具有对应数量参数的invoke方法

比Java8更接近的函数式/声明式范式

Lambda表达式存在三个简化约定:

  1. 如果lambda表达式,参数类型为空,可以省略参数和箭头
  2. 如果lambda表达式参数类型只有一个,可以省略参数和箭头并使用it作为形参
  3. 如果lambda表达式作为函数的最后一个参数,可以把放到括号外

来一段代码实例

现在有一个需求:添加前缀,分隔符,后缀,打印出某个给定的集合里的所有元素

java实现:

public class SeparatorUtils {

    /**
     *
     * @param collections 集合
     * @param prefix 前缀
     * @param separator 分隔符
     * @param postfix 后缀
     * @param <T> 集合泛型
     * @return 分隔后的结果
     */
    @NotNull
    public static <T> String separate(Collection<T> collections, String prefix , String separator, String postfix) {
        Objects.requireNonNull(collections);
        StringBuilder stringBuilder = new StringBuilder();
        stringBuilder.append(prefix);
        int index = 0;
        for (T t : collections) {
            if(index > 0){
                stringBuilder.append(separator);
            }
            stringBuilder.append(t.toString());
            index ++;
        }
        stringBuilder.append(postfix);
        return stringBuilder.toString();
    }
}

对应Kotlin实现:

@JvmOverloads
public fun <T> separate(
        collection: Collection<T>,
        prefix: String = "(",
        separator: String = ",",
        postfix: String = ")"
): String {
    val stringBuilder = StringBuffer(prefix)
    for ((index, element) in collection.withIndex()) {
        if (index > 0) {
            stringBuilder.append(separator)
        }
        stringBuilder.append(element)
    }
    stringBuilder.append(postfix)
    return stringBuilder.toString()
}

对上述代码作简化,如下

@JvmOverloads
fun <T> separate3(
        collection: Collection<T>,
        prefix: String = "(",
        separator: String = ",",
        postfix: String = ")"
): String {
    return StringBuffer(prefix)
            .apply {
                for ((index, element) in collection.withIndex()) {
                    if (index > 0) {
                        append(separator)
                    }
                    append(element)
                }
                append(postfix)
            }.toString()
}

使用拓展函数,减少一个入参

fun <T> Collection<T>.separateInto(prefix: String = "(",
                                   separator: String = ",",
                                   postfix: String = ")"
): String {
    return separate3(this, prefix, separator, postfix)
}

Kotlin中闭包的实现

一个简易的方法调用次数计数器:

fun compute(): () ->Int{
    var count = 0
    fun inner(): Int{
        count ++
        return count
    }
    return {inner()}
}

调用

val message:() -> Int = compute()
    for(i in 0..3){
        println(message())
    }

输出结果

1234

此时,compute()方法里的count像全局变量,累加记数。简易的访问流程:外部的A访问B函数,B内部的函数C访问B,同时B返回C。

闭包就是这样一种结构:函数嵌套一个访问自己变量的内部函数结构

闭包带给我们两个好处:

  1. 让某个变量保存在内存里,能起到消除不与其他方法通信的成员变量的作用
  2. 外部能访问到函数内部的变量

相关链接

Kotlin学习相关链接:

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,290评论 4 363
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,399评论 1 294
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,021评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,034评论 0 207
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,412评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,651评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,902评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,605评论 0 199
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,339评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,586评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,076评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,400评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,060评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,083评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,851评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,685评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,595评论 2 270

推荐阅读更多精彩内容

  • 一人,两年,承的是最磨人却美好纯粹的时光。 一人,两别,记住的只有依稀的笑颜和绕耳的言语。 一人,两手,牵着我远离...
    海里的水阅读 283评论 0 1
  • 属于创建型模式,它提供了一种创建对象的最佳方式,在任何需要生成复杂对象的地方,都可以使用工厂方法模式,即用工厂方法...
    龙遁流阅读 268评论 0 0
  • 江雪 柳宗元 千山鸟飞绝,万径人踪灭。 孤舟蓑笠翁,独钓寒江雪。 说到写雪的诗,这一首,应该是最负盛名。不需要读诗...
    李大菠萝阅读 780评论 1 0