Kotlin基础语法


定义包

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)

文章作者: Xyq
版权声明: 本博客所有文章除特別声明外,均采用 CC BY 4.0 许可协议。转载请注明来源 Xyq !
  目录