定义包
package my.demo
import java.util.*
定义函数
带有两个 Int 参数、返回 Int 的函数:
fun sum(a: Int, b: Int): Int {
	return a + b
}
将表达式作为函数体、返回值类型自动推断的函数
fun sum(a: Int, b: Int) = a + b
函数返回无意义的值
fun printSum(a: Int, b: Int):Unit{
  	//$varName 表示变量值,${表达式}执行表达式
	println("sum of $a and $b is ${a + b}")
}
Unit返回类型可以省略
fun printSum(a: Int, b: Int):Unit{
	println("sum of $a and $b is ${a + b}")
}
可变长参数函数,用 vararg 关键字进行标识:
函数的变长参数可以用 vararg 关键字进行标识:
fun vars(vararg v: Int) {
	for(it in v) {
    	print(it)
	}
}
vars(1,2,3,4) ,输出1234
定义变量
定义只读局部变量使用关键字 val 定义。只能为其赋值一次。
val a: Int = 1 //立即赋值
val b = 3.0 //自动推断类型为Short
val c: Double //变量申明
c = 2.9D //赋值
// Byte 定义字节值
var b:byte = 127
// Short 定义短整型值
var number:Short = 32767
// Int 定义Int值
var age: Int = 2017 或者 var age = 2017 (自动推断类型)
// Long 定义长整型
var money: Long = 9999999999L 或者 var money = 99999999999L 
// Float 定义单精度值
var f: Float = 123456789f 或者 var f = 123456789f
// Double 定义双精度值
var d: Double = 66666666.0 或者 var d = 66666666.0
// boolean 定义布尔值
var biu = true/false
// Char 定义字符
var cili = "c"
定义重复赋值变量使用var 关键字
var a: Int = 32
a = 2
延迟初始化属性 (Late-Initialized Properties)
在Kotlin中,声明为具有非空类型的属性必须在构造函数中初始化,但是往往不希望在构造函数中初始化,例如在通过依赖注入或单元测试的设置方法来初始化属性的时候,不能在构造器中提供一个非空的初始化语句,为了处理这种情况,就要在属性上加lateinit关键字来延迟初始化
public class MyTest {
    lateinit var subject: TestSubject
    @Before fun setup() {
        subject = TestSubject()
    }
    @Test fun test() {
        subject.method() 
    }
}
注:lateinit只能够在var类型的属性中,不能用于构造函数,而且属性不能有自定义的getter和setting,这些属性必须是非空类型,并且不能是基本类型。 如果在一个延迟初始化的属性初始化前调用,会导致一个特定异常,调用的时候值还没有初始化.
NULL检查机制
Kotlin的空安全设计对于声明可为空的参数,在使用时要进行空判断处理,有两种处理方式,字段后加!!像Java一样抛出空异常,另一种字段后加?可不做处理返回值为 null或配合?:做空判断处理
//类型后面加?表示可为空
var age: String? = "23" 
//抛出空指针异常
val ages = age!!.toInt()
//不做处理返回 null
val ages1 = age?.toInt()
//age为空返回-1
val ages2 = age?.toInt() ?: -1
类型检测及自动转换
fun getStringLength(obj: Any): Int? {
  if (obj !is String)
    return null
  // 在这个分支中, `obj` 的类型会被自动转换为 `String`
  return obj.length
}
基本数据类型
Kotlin 的基本数值类型包括 Byte、Short、Int、Long、Float、Double 等
| 类型 | 位宽度 | 示例 | 
|---|---|---|
| Double | 64 | 2.0 | 
| Float | 32 | 2f | 
| Long | 64 | 3L | 
| Int | 32 | 3 | 
| Short | 16 | 4 | 
| Byte | 8 | 5 | 
可以使用下划线使数字常量更易读
val oneMillion = 1_000_000
val hexBytes = 0xFF_EC_DE_5E
val phoneNumber = 185_5162_3845
在 Kotlin 中,三个等号 === 表示比较对象地址,两个 == 表示比较两个值大小。
val a: Int = 10000
//经过了装箱,创建了两个不同的对象
val boxedA: Int? = a
val anotherBoxedA: Int? = a
//虽然经过了装箱,但是值是相等的
println(boxedA === anotherBoxedA) //  false,值相等,对象地址不一样
println(boxedA == anotherBoxedA) // true,值相等
类型转换:每种数据类型都有下面的这些方法,可以转化为其它的类型:
toByte(): Byte
toShort(): Short
toInt(): Int
toLong(): Long
toFloat(): Float
toDouble(): Double
toChar(): Char
从kotlin1.3开始,引入了无符号整型数据类型,目前还是实验性的。
| 类型 | 位宽度 | 范围 | 示例 | 
|---|---|---|---|
| UByte | 8 | 0~255 | 221u | 
| UShort | 16 | 0~2^16^-1 | 12u | 
| UInt | 32 | 0~2^32^-1 | 36u | 
| ULong | 64 | 0~2^64^-1 | 1284uL | 
kotlin修饰符及关键字
public / protected / private / internal
expect / actual
final / open / abstract / sealed / const
external
override
lateinit
tailrec
vararg
suspend
inner
enum / annotation
companion
inline
infix
operator
data
与Java不同的是,kotlin默认的访问修饰符是public,而且多了一个internal(模块)修饰符
其他private 和 protected的作用范围和Java一样
- private:只该类可见(包括该类的所有成员)
 - protected:同private 加子类可见
 - internal:在同一个模块中可见
 - public:公共,所有都可见
 
在kotlin中增加了top-level属性和方法,与class同级
package foo.bar
val prop: String = "top-level-prop"
fun demo() {
    loge("top-level", "top-level-demo()")
}
class Kot {
    fun v() {
        loge("top-level", prop)
        demo()
    }
}
上面的prop属性和demo()方法就是顶级属性及方法,在Kot类编译成class文件的时候,会把Top-level的属性和函数创建到以类名+Kt为名的class文件中,如:KotKt.class
顶级属性和方法前面不能用protected访问修饰符
Kotlin随机数
kotlin1.3加入多平台random随机数,kotlin.random.Random
Random.nextInt() //Int.MIN_VALUE ~ Int.MAX_VALUE
Random.nextInt(52) //0 ~ 52
Random.nextBoolean() //true or false
Random.nextFloat() //0 ~ 1
Random.nextDouble() //0 ~ 1
Random.nextDouble(5.0) //0 ~ 5.0
//生成一个长度23的随机数组
val nextBytes = Random.nextBytes(23)
for (it in nextBytes) {
	println(it)
}
运算符
单目运算符
| 表达式 | 对应函数 | 
|---|---|
| +a | a.unaryPlus() | 
| -a | a.unaryMinus() | 
| in | a.not() | 
| a++ | a.inc() | 
| a– | a.dec() | 
双目运算符
| 表达式 | 对应函数 | 
|---|---|
| a+b | a.plus(b) | 
| a-b | a.minus(b) | 
| a*b | a.times(b) | 
| a/b | a.div(b) | 
| a%b | a.rem(b) / a.mod(b) 已过时 | 
| a..b | a.rangTo(b) | 
| a in b | a.contains(b) | 
| a !in b | !a.contains(b) | 
| a += b | a.plusAssign(b) | 
| a -= b | a.minusAssign(b) | 
| a *= b | a.timesAssign(b) | 
| a /= b | a.divAssign(b) | 
| a %= b | a.remAssign(b) | 
| a && b | a.and(b) | 
| a || b | a.or(b) |