1. 程式人生 > >kotlin 練習

kotlin 練習

kotlin基礎語法

96  samychen  2017.05.28 17:07* 字數 1224 閱讀 2434評論 0

每種程式語言都有一定的語法、語義和執行順序(同步),學習一種新語言也都是從這三者出發,下面我們就只針對kotlin的語法來做簡單的介紹。

Kotlin有自己的特性不該被Java的思維所束縛。

  1. 基本語法準則:
    在Kotlin中常量用 val 宣告,變數用 var 宣告;
    關鍵字在前面,型別以冒號 :隔開在後面,也可以省略直接賦值;
    型別後帶問號 ? 表示可為空型別(預設空型別安全);
    常量 val 延遲載入 by lazy{} ;
    預設是執行緒安全的,關閉執行緒安全 lazy(LazyThreadSafetyMode.NONE){} ;
    變數 var 延遲載入 lateinit ;
    內部類和引數預設為public,而在Java中為private
    類預設為不可繼承(final),想要可被繼承要宣告為 open 或 abstract
    取消了static關鍵字,靜態方法和引數統一寫在 companion object 塊
    internal模組內可見,inner內部類
//常量陣列int[][][] arrs = new int[3][2][1];
val arrs = Array(3) { Array(2) { IntArray(1) } }
internal var name: String? = null//型別後帶問號 ? 表示可為空型別(預設空安全) internal var age: Int = 0//internal模組內可見,inner內部類 //當我們只有單個構造器時,我們需要在從父類繼承下來的構造器中指定需要的引數。這是用來替換Java中的super呼叫的。 open class Animal(name: String) class Person(name: String, surname: String) : Animal(name) 

kotlin是空型別安全的,所有變數預設為"not null",必須顯式在型別後新增?修飾符才可賦值為null。

var notNullArtist: Artist = null//編譯不通過,因為notNullArtist不能為null
var artist: Artist? = null//編譯通過 artist.print()//編譯不通過,因為artist可能為空 /** Kotlin進行空判斷處理,有兩種處理方式: * 1. 丟擲空異常,欄位後加 !! * 2. 不做處理直接跳過,欄位後加 ? */ artist?.print()//編譯通過,做了非空判斷,只有當artist!=null時才呼叫print() artist!!.print()//這種用法只有在確認artist不為null時才能呼叫,否則丟擲空指標異常 val name = artist?.name?:"empty"//當artist為null時可以指定一個預設值 
  1. 條件語句
    if...else 正常使用,不過移除了 switch 用更強大的 when 替代,when子式可以是各種返回Boolean的表示式
val x = 7
when (x) {
  in 1..5 -> print("x is in the range")
  in validNumbers -> print("x is valid") !in 10..20 -> print("x is outside the range") else -> print("none of the above") } 

kotlin儘可能多的使用when

  1. 迴圈語句
    while 和 do...while 同Java並無區別, for 則有很大改變並多出了幾個變種
fun main(args: Array<String>) {
    var list = ArrayList<String>()
    add(list)
    list.forEachIndexed { i, s -> print(list[i]) print(s) } println() //如果沒有指定函式的返回值,它就會返回Unit,與Java中的void類似,但是Unit是一個真正的物件。當然也可以指定任何其它的返回型別: list.forEachIndexed(object :(Int,String) -> Unit{ override fun invoke(i: Int, s: String) { print(list[i]) print(s) } }) } //遞增for (int i = 0; i < list.size(); i++) for (i in list.indices) { print(list[i]) } //遞增for (int i = 2; i < list.size(); i++) for (i in 2..list.size-1) { print(list[i]) } //遞減for (int i = list.size(); i >= 0; i--) for (i in list.size downTo 0) { print(list[i]) } //操作列表內的物件 for (item in list) { print(item) } //加強版 for((i,item) in list.withIndex()){ print(list[i]) print(item) } //變種版 list.forEach { print(it) } list.forEachIndexed { i, s -> print(list[i]) print(s) } list.forEachIndexed(object :(Int,String) -> Unit{ override fun invoke(i: Int, s: String) { print(list[i]) print(s) } }) fun add(list:MutableList<String>) { for (i in 0..4) { list.add(i.toString() + "") } } 

冒號使用
  在Kotlin中冒號 : 用萬能來稱呼絕不為過。常量變數的型別宣告,函式的返回值,類的繼承都需要它

除此之外還有一個特別的地方也需要它,使用Java類的時候。Kotlin最終會還是編譯成Java位元組碼,使用到Java類是必然的,在Kotlin語法如下
val intent = Intent(this, MainActivity::class.java)

指定上下文的@

除了冒號另一個重要符號 @ ,java程式碼中經常用到內部類和匿名內部類,有時我們不能確定this指代的上下文,Java可以使用XXX.this指代具體上下文,在kotlin中的做法是[email protected]

class User {
    inner class State{ fun getUser(): User{ //返回User return this@User } fun getState(): State{ //返回State return this@State } } } 

kotlin的特色

Java的 getter/setter 方法自動轉換成屬性,對應到Kotlin屬性的呼叫

public class User {
    private String name; private String age; public String getName() { return name; } public void setName(String name) { this.name = name; } public String getAge() { return age; } public void setAge(String age) { this.age = age; } } 

對應的kotlin

val user = User()
//賦值
user.name = "tutu"
user.age = "23"
//取值
val name = user.name
val age = user.age

class User { var name: String? = null var age: String? = null } 

有時 getter/setter 方法比較複雜,這就需要自定義 getter/setter 了,實現一個Java中常用的單例,這裡只為了展示,單例在Kotlin有更簡單的方法實現,只要在 package 級別建立一個 object 即可

class User {
    companion object {//靜態方法和引數統一寫在 companion object 塊
         //volatile不保證原子操作,所以,很容易讀到髒資料。在兩個或者更多的執行緒訪問的成員變數上使用volatile @Volatile var instance: User? = null get() { if (field == null) { synchronized(User::class.java) { if (field == null) field = User() } } return field } } var name: String? = null var age: String? = null } 

自定義 getter/setter 重點在 field ,跟我們熟悉所Java的 this 指代當前類一樣, field 指代當前引數,直接使用引數名 instance 代替不會報錯但單例就沒效果了

字串問題

在Java中拼接字串的程式碼可讀性都很差,在Kotlin字串拼接變得非常簡潔,只需用 $ 後面加上引數名,複雜的引數要加上 {}

 val pair = Pair(1, "one")
 val (num, name) = pair
 println("num = $num, name = $name")

輸出num = 1, name = one

Java8新特性lambda的支援

lambda需要一個函式,但是又不想費神去命名一個函式的場合下使用,也就是指匿名函式。使用功能介面,把介面名、方法名和引數型別省掉不寫再加個 -> 罷了。

使用Java開發Android時,處理監聽回撥是常見的事,kotlin可以直接編寫監聽回撥而不用再通過匿名物件傳遞onClick方法,這個特性被稱為Lambda表示式

view.setOnclickListener({
      Toast.makeText(this, "Hello World!", Toast.LENGTH_LONG).show()
})

擴充套件函式

可以為任何已經存在的類新增新函式,相比傳統工具類,擴充套件函式更具有可讀性。

//為Fragment新增擴充套件函式
fun Fragment.toast(message: CharSequence, duration: Int = Toast.LENGTH_LONG){
      Toast.makeText(getActivity(), message, duration).show()
}

呼叫時直接呼叫fragment.toast("Hello World!")或fragment.toast("Hello World!", 2000)

Kotlin中的引數與Java中有些不同。如你所見,我們先寫引數的名字再寫它的型別。上面呼叫的第二個引數(length)指定了一個預設值。這意味著你呼叫的時候可以傳入第二個值或者不傳,這樣可以避免你需要的過載函式。

函式式支援(lambda),函式是一級公民

集合操作

  list轉map(associateBy)

  場景:訂單列表轉換成以 id為key 的訂單map

val mainOrders = orderDao!!.queryUserOrder(param)
val orderMap = mainOrders.associateBy { it.id }.toMap()

  map的key或者value轉換

  假如一個map的key是String,需要轉換成Long;或者map的value是一個物件,要轉成另一個物件。按照標準Java寫法,可以要new一個新的map,然後迴圈老的map,在kotlin中,一行程式碼搞定

val map = mutableMapOf(1 to 1, 2 to 2)
val newMap = map.mapKeys { "key_${it.key}" }.mapValues { "value_${it.value}" }
println(newMap)
//列印結果 {key_1=value_1, key_2=value_2}
val pair = Pair("ss","sg") val map = mapOf(pair) val map1=map.mapKeys { entry -> "${entry.value}!" } for((key,value) in map1){ println("map1:key=$key") println("map1:value=$value") } val map2 =map.mapKeys { (key, value) -> "$value" } for((key,value) in map2){ println("map2:key=$key") println("map2:value=$value") } val map3=map.mapValues { entry -> "${entry.value}!" } for((key,value) in map3){ println("map3:key=$key") println("map3:value=$value") } val map4=map.mapValues { (key, value) -> "$value" } for((key,value) in map4){ println("map4:key=$key") println("map4:value=$value") } 列印結果: map1:key=sg! map1:value=sg map2:key=sg map2:value=sg map3:key=ss map3:value=sg! map4:key=ss map4:value=sg 

參考文獻
  Kotlin for android Developers
  kotlin 指令碼練習