那些年Kotlin轻易能实现,java很难实现的语法糖

做android开发那么多年, 一直都是兢兢业业用java去码代码,直到两年前的一天发现了kotlin这门神奇的语言。犀利的语法特性,优雅的表达方式,从此掉入此坑。

这里有个对比链接可以看到表达同一个含义,kotlin相对java在表达上的优势。

最近google io大会把kotlin定为Android开发第一级语言,着实掀起了一股讨论kotlin的浪潮。那么,未来的androider是不是会用kotlin替代java呢?这里我们来讨论下一个话题:那些Kotlin轻易能实现,java很难实现的语法糖,由此来看一看kotlin的独特魅力。

函数编程

在kotlin中,函数和类一样是一等公民,函数也可以作为参数传递(高阶函数特性),且可执行(闭包特性)。举个栗子

// 函数作为对象
val magic = fun(name: String) {
    println("hello $name")
}

// 函数闭包执行
magic("tom")


// 函数作为参数
fun test(funParam: (String) -> Unit) {
    funParam("tom2")
}

test(magic) 

在集合类的数据处理上,kotlin中的filtermap等能方便做到数据转换。举个栗子,给定一个数字数组,过滤出数字为3的倍数的数字集合,数字前加#,以逗号分隔输出。如数组为0~10, 打印输出#0, #3, #6, #9

java代码如下

private static String solveList(List list) {
    // filter
    List filterResult = new ArrayList<>();
    for (Integer i : list) {
        if (i % 3 == 0) {
            filterResult.add(i);
        }
    }

    // map
    List mapResult = new ArrayList<>();
    for (Integer i : filterResult) {
        mapResult.add("#" + i);
    }

    // 逗号分隔
    StringBuilder sb = new StringBuilder();
    for (String s : mapResult) {
        sb.append(s).append(", ");
    }

    if (sb.length() > 0) {
        return sb.substring(0, sb.length() - 2);
    } else {
        return "";
    }
}

kotlin代码如下,异常的简洁。

fun solveList(list: List) : String {
    return list.filter { it % 3 == 0 }.map { "#$it" }.reduce { s1, s2 -> "$s1, $s2" }
}

kotlin用的是函数式思维方式,函数式三板斧filter, map, reduce 接受函数作为参数,屏蔽了内部实现。 比如说这里3的倍数改成5的倍数,kotlin只要改filter的参数(函数参数)即可,java则要改方法的内部实现。

Extension

kotlin可对系统类进行方法扩展,如可以为集合增加swap函数功能

fun  MutableList.swap(index1: Int, index2: Int) {
    val tmp = this[index1] // 'this' corresponds to the list
    this[index1] = this[index2]
    this[index2] = tmp
}

定义之后MutableList类型就可以直接调用swap方法

val list = mutableListOf("aa", "bb")
list.swap(0, 1)

这种形式java是无法做到的,java要实现同样的功能必须额外定义一个新类。(事实上kotlin extension原理也是编译器生成一个新类~~)

数据类

数据类大量重复的getter和setter相信会是很多人在开发过程中吐槽的一个点。举一个很经典的例子,我们需要一个Person的数据类。

public class Person {
    private String name;
    private int age;
    private int sex;
    private float height;

    public Person(String name, int age, int sex, float height) {
        this.name = name;
        this.sex = sex;
        this.age = age;
        this.height = height;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    public int getSex() {
         return sex;
    }

    public void setSex(int sex) {
        this.sex = sex;
    }

    public float getHeight() {
        return height;
    }

    public void setHeight(float height) {
        this.height = height;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", sex=" + (sex == 0 ? "男" : "女") +
                ", height=" + height +
                "}";
    }
}

在Kotlin里,我们只需要一行代码就能完成以上的功能:

data class Person(var name: String, var age: Int, var sex: Int, var height: Float)

data class有效的提供了一种构建器模式,可以指定参数初始化,同时默认提供copy方法

//创建对象
val person: Person = Person(name="tom", age = 10, height = 1.7f)

Type alias

java没alias的说法, kotlin type alias暂时很少用到,就不细讲了,有兴趣的参考官网链接 type-aliases

字符串模板

kotlin通过${expression}输出字符串,expression表示任意表达式 如

println("My name's {user.name}")
println("1 + 2 = ${1 + 2}")

java用String.format形式实现

kotlin输出list, map等类型就友好的多。

val list = (0..10).toMutableList()
println("list=$list")
// 输出list=[0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10]

val map = mapOf(1 to "One", 2 to "Two", 3 to "Three")
println("map=$map")

// 输出map={1=One, 2=Two, 3=Three}

option参数

kotlin有option参数的说法,如定义个图片加载接口,java需要通过重载方式定义

interface IHJImageLoader {
    public void displayImage(String uri, ImageView imageView);

    public void displayImage(String uri, ImageView imageView, HJImageLoaderOptiono option);

    public void displayImage(String uri, ImageView imageView, HJImageLoaderListener loaderListener);

    public void displayImage(String uri, ImageView imageView, HJImageLoaderOptiono option, HJImageLoaderListener loaderListener);
}

而kotlin只要定义个一个方法就行

interface IHJImageLoader {
    fun displayImage(uri: String?, imageView: ImageView?, option: HJImageLoaderOption? = HJImageLoaderOption.defaultOption(), loaderListener: HJImageLoaderListener? = null)
}

调用方式都是类似

val imageLoaderImpl: IHJImageLoader = ...

imageLoaderImpl.displayImage(uri, imageView)
imageLoaderImpl.displayImage(uri, imageView, option)
imageLoaderImpl.displayImage(uri, imageView, option, loaderListener)

操作符重载

kotlin操作符重载在DSL中得以发扬光大, 如下面一段语句表达

html {
    body {
        div {
            a("http://kotlinlang.org") {
                target = ATarget.blank
                +"Main site"
            }
        }
    }
}

java要表达同样的含义,至少得html, body, div, a对象都new一遍,且要设置成员变量。冗余度和可读性上都远不如上面的表达来的舒爽。

协程编程

todo

你可能感兴趣的:(那些年Kotlin轻易能实现,java很难实现的语法糖)