十分钟过一遍Kotlin知识点

这是团队内部分享的Kotlin的知识点,还在不断完善的阶段。团队用Kotlin作为后端开发语言已经一年多,知识点还在不断的挖掘规范中。期待和更多的喜欢Kotlin的同学们一起讨论。

1:变量声明

val:只读变量,可以之后初始化,不过如果未初始化之前该变量无法使用,编译器会报错,编译成字节码的时候编译器应该会做优化,不建议之后初始化,什么时候用到什么时候定义。

Var:可变(略)

2:类型推导

kotlin虽然是一种强类型的jvm语言,一定程度上保证了类型安全,但是基于好的编码习惯,除非特别明显类似于val a = Example()这种,其他尽量不要省略类型,尤其一些函数,省略返回类型有时候可能会出现问题。

3:Array

case 1: val a:Array = arrayof(1,2,3)

case 2:val b:Array = Array(3,)

4:import renaming

import renaming可以使用,注意到是,如果重命名后,只能使用重命名之后的声明,和正常的import一样。

另外,有意思的是你可以重定义和重命名相同的名字,不过该文件内会优先选择import的声明。

5:range

a:闭区间(..)for:val range:IntRange= 1 .. 5 包括1和5

b:半开半闭区间(until)for:val range : IntRange = 1 until 5 包括1不包括5

6:异常

不同于Java,kotlin中所有的异常都是unchecked异常。

Checked异常:需要在声明中指出。

Unchec异常:不需要在声明中指出。

7:引用相等和结构相等

结构相等(==):默认会调用equal方法(如果不重写equal方法,Any默认的equal方法好像是判断引用是否相等)

引用相等(===):判断饮用指向的是否是同一实例

注意:结构相等对于null也是安全的,编译器会作null检查,如果都是null则返回true。

8:访问修饰符(默认public)

public:默认修饰符,被其修饰的在任何位置都能访问

private:只能在当前源文件内使用

internal:在同一模块内使用

protected:无效修饰符,只用于类和接口内

9:控制流作表达式

不同于Java,if .. else /try.. catch在kotlin中都是表达式,表达式意味着控制流可以直接作为右值。

10:自动类型转换

11:显示类型转换(as)

as转换失败会跑异常,所以使用as的时候要注意,这里可能会抛异常,如果不想抛异常,可以使用is,另外,as转换后可能会返回null,赋值编译器并不会帮你做检查,所以coding过程中要注意。Case如下:

12:when

when(arg):有点类似于穷举,判断当前arg处于哪个分支,在这里不要想着做所谓的bool判断,不支持,可以针对arg做bool判断,比如用in。

When:分支全是bool判断。

13:类型结构

Any:类似于Java的Objevt

Unit:不同于void,Unit也是一个实实在在的类型,只不过是singleton instane。

Nothing:所有类的子类。

14:构造函数

主构造函数:val/var可要可不要,如果不要则默认没有getter/setter方法,这块我觉得如果没有val/var其实就是传了个参数过去,class中并没有这个字段。

非主构造函数:val/var必须不要,一定要调用主构造函数。

15:初始化顺序(这里不考虑实力化过程中内存分配时的初始化,只是从kotlin语法上考究)

字段初始化赋初值

init

构造函数

与Java构造代码块不同的是,init有点类似于pre constructor,不比在乎字段是在init之前或者之后。

16:嵌套类

inner class:和外部类有关联,需要依赖于一个外部类实力,外部类一切对其可见。

static class:和外部类没什么关联,只是定义在外部类内部。

17:object /function/companion

其本质都是生成一个final类,其方法都被标注为static。

备注:这块没怎么研究,有时间可以看看字节码。

18:代理(by)

19:密闭类(可以理解成是类型的枚举,使用时需要该类型的实例)

20:local functions(外部不可访问,作用域内变量对其都可见)

21:函数默认参数

重写的时候必须完整的函数签名,且重写函数不允许再次指定参数默认值

22:扩展函数(成员扩展函数可以被重写,没什么需要特别注意的地方)

23:函数字面量(值得注意的是如果参数可以推导出来可以省略)

24:尾递归

如果函数定义中用到了递归,且返回值就是其递归调用,则可以采用尾递归,编译器会自动优化调用过程,采用迭代方式执行该调用,不至于产生那么多递归栈。

25:varargs

如果该关键字标识的参数不是最后一个,则之后的参数要采用命名参数传递实参。

26:Spread operator

27:Apply

28:Let

29:Run

30:Lazy

31:Use

32:Repeat(从0开始)

33:require/assert/check

项目中这几个用的很少,一般都定义自己的require/assert/check进行相应的抛异常和处理操作。

34:SAM(TODO)

35:top-level functions

编译器根据包名生成和包名同名的class,所有的函数都以static形式成为该class的成员函数。

36:匿名函数

37:函数引用

38:typealias

39:属性

同Java初始化构造快,kotlin init用到的属性必须要先定义,否则无法在init块中使用。

40:延迟初始化

不能是基本类型

不能自定义getter/setter

如果未赋值使用的话会抱异常

41:代理属性(TODO)

42:lazy initialization

用法有三种,详情都在上面,细节可以参考详细介绍,有时间可以仔细钻研一下。

43:lazy versuslateinit

lazy只能用于val变量,lateinit只能用于var变量

lateinit不能用于null

44:Observables

用来监控属性变化和对属性进行约束

45:not-nullproperty delegate

感觉功能有点类似于lateinit

46:as[?]

不安全,转换不成功会抛异常,可以用as?,安全转换,如果转换不成功会返回null

47:Optional

不用多说

48:Upper Bounds

classUpperBoundsTestwhereT:Comparable,T:Serializable,V:Serializable,V:Comparable

49:out/in

in用法省略,和out相同,继承语义和out正好相反。

50:未完待续

编码规约建议

kotlin非空断言(!!)尽量不要使用。

最好不要出现warning,如果有最好处理掉。

能用枚举的话最好用枚举,不要习惯使用字面量。

能用全局变量就用全局变量,最好不要全都用字面量。比如如果频繁的出现1000、10000可以用全局变量千/万来替代,可读性会更好。

建议使用统一的命名规范,比如grpc的实现采用统一的命名规则,模型层采用和数据表同名的方式,以便迅速定位哪个类属于数据表的模型实现,其他的业务也如此,便于查找相关联的业务。

  • 发表于:
  • 原文链接https://kuaibao.qq.com/s/20180706B0K8BN00?refer=cp_1026
  • 腾讯「云+社区」是腾讯内容开放平台帐号(企鹅号)传播渠道之一,根据《腾讯内容开放平台服务协议》转载发布内容。
  • 如有侵权,请联系 yunjia_community@tencent.com 删除。

扫码关注云+社区

领取腾讯云代金券