快速上手 Kotlin 的 11 招
本文轉載自 Kotlin 公眾號(KotlinX)
作者:bennyhuo
最近經常會收到一些 “用 Kotlin 怎麼寫” 的問題,作為有經驗的程式設計師,我們已經掌握了一門或者多門語言,那麼學 Kotlin 的時候就經常會有類似 “ ‘再見’用日語怎麼說?”、“ ‘你好’ 用西班牙語怎麼說?” 的問題,所以我決定把一些常用的語法對照列舉出來,如果大家熟悉 Java,那麼快速上手 Kotlin 會變得非常地容易。
這篇文章主要是寫給需要快速上手 Kotlin 的 Java 程式設計師看的,這時候他們關注的是如何 Kotlin 寫出類似某些 Java 的寫法,所以本文基本不涉及 Kotlin 的高階特性。
1. 如何定義變數
Java 定義變數的寫法:
String string = "Hello";
基本等價的 Kotlin 定義變數的寫法:
var string: String = "Hello"
Java 定義 final 變數的寫法:
final String string = "Hello";
注意到前面的是一個編譯期常量,Kotlin 當中應該這麼寫:
const val string: String = "Hello"
同樣是 final 變數,Java 這麼寫:
final String string = getString();
注意到,這個不是編譯期常量,Kotlin 這麼寫:
val string: String = getString()
另外, Kotlin 有型別推導的特性,因此上述變數定義基本上都可以省略掉型別 String。
2. 如何定義函式
Java 當中如何定義函式,也就是方法,需要定義到一個類當中:
public boolean testString(String name){
...
}
等價的 Kotlin 寫法:
fun testString(name: String): Boolean {
...
}
注意到返回值的位置放到了引數之後。
3. 如何定義靜態變數、方法
Java 的靜態方法或者變數只需要加一個 static 即可:
public class Singleton{
private static Singleton instance = ...;
public static Singleton getInstance(){
...
return instance;
}
}
用 Kotlin 直譯過來就是:
class KotlinSingleton{
companion object{
private val kotlinSingleton = KotlinSingleton()
@JvmStatic
fun getInstance() = kotlinSingleton
}
}
注意 getInstance 的寫法。 JvmStatic 這個註解會將 getInstance 這個方法編譯成與 Java 的靜態方法一樣的簽名,如果不加這個註解,Java 當中無法像呼叫 Java 靜態方法那樣呼叫這個方法。
另外,對於靜態方法、變數的場景,在 Kotlin 當中建議使用包級函式。
4. 如何定義陣列
Java 的陣列非常簡單,當然也有些抽象,畢竟是編譯期生成的類:
String[] names = new String[]{"Kyo", "Ryu", "Iory"};
String[] emptyStrings = new String[10];
Kotlin 的陣列其實更真實一些,看上去更讓人容易理解:
val names: Array<String> = arrayOf("Kyo", "Ryu", "Iory")
val emptyStrings: Array<String?> = arrayOfNulls(10)
注意到,Array T 即陣列元素的型別。另外,String? 表示可以為 null 的 String 型別。
陣列的使用基本一致。需要注意的是,為了避免裝箱和拆箱的開銷,Kotlin 對基本型別包括 Int、Short、Byte、Long、Float、Double、Char 等基本型別提供了定製版陣列型別,寫法為 XArray,例如 Int 的定製版陣列為 IntArray,如果我們要定義一個整型陣列,寫法如下:
val ints = intArrayOf(1, 3, 5)
5. 如何寫變長引數
Java 的變長引數寫法如下:
void hello(String... names){
...
}
Kotlin 的變長引數寫法如下:
fun hello(vararg names: String){
}
6. 如何寫三元運算子
Java 可以寫三元運算子:
int code = isSuccessfully? 200: 400;
很多人抱怨 Kotlin 為什麼沒有這個運算子。。。據說是因為 Kotlin 當中 : 使用的場景比 Java 複雜得多,因此如果加上這個三元運算子的話,會給語法解析器帶來較多的麻煩,Scala 也是類似的情況。那麼這中情況下,我們用 Kotlin 該怎麼寫呢?
int code = if(isSuccessfully) 200 else 400
注意到,if else 這樣的語句也是表示式,這一點與 Java 不同。
7. 如何寫 main 函式
Java 的寫法只有一種:
class Main{
public static void main(String... args){
...
}
}
注意到引數可以是變長引數或者陣列,這二者都可。
對應 Kotlin,main 函式的寫法如下:
class KotlinMain{
companion object{
@JvmStatic
fun main(args: Array<String>) {
}
}
}
Kotlin 可以有包級函式,因此我們並不需要宣告一個類來包裝 main 函式:
fun main(args: Array<String>){
...
}
8. 如何例項化類
Java 和 C++ 這樣的語言,在構造物件的時候經常需要用到 new 這個關鍵字,比如:
Date date = new Date();
Kotlin 構造物件時,不需要 new 這個關鍵字,所以上述寫法等價於:
val date = Date()
9. 如何寫 Getter 和 Setter 方法
Java 的 Getter 和 Setter 是一種約定俗稱,而不是語法特性,所以定義起來相對自由:
public class GetterAndSetter{
private int x = 0;
public int getX() {
return x;
}
public void setX(int x) {
this.x = x;
}
}
Kotlin 是有屬性的:
class KotlinGetterAndSetter{
var x: Int = 0
set(value) { field = value }
get() = field
}
注意看到,我們為 x 顯式定義了 getter 和 setter,field 是 x 背後真正的變數,所以 setter 當中其實就是為 field 賦值,而 getter 則是返回 field。如果你想要對 x 的訪問做控制,那麼你就可以通過自定義 getter 和 setter 來實現了:
class KotlinGetterAndSetter{
var x: Int = 0
set(value) {
val date = Calendar.getInstance().apply {
set(2017, 2, 18)
}
if(System.currentTimeMillis() < date.timeInMillis){
println("Cannot be set before 2017.3.18")
}else{
field = value
}
}
get(){
println("Get field x: $field")
return field
}
}
10. 如何延遲初始化成員變數
Java 定義的類成員變數如果不初始化,那麼基本型別被初始化為其預設值,比如 int 初始化為 0,boolean 初始化為 false,非基本型別的成員則會被初始化為 null。
public class Hello{
private String name;
}
類似的程式碼在 Kotlin 當中直譯為:
class Hello{
private var name: String? = null
}
使用了可空型別,副作用就是後面每次你想要用 name 的時候,都需要判斷其是否為 null。如果不使用可控型別,需要加 lateinit 關鍵字:
class Hello{
private lateinit var name: String
}
lateinit 是用來告訴編譯器,name 這個變數後續會妥善處置的。
對於 final 的成員變數,Java 要求它們必須在構造方法或者構造塊當中對他們進行初始化:
public class Hello{
private final String name = "Peter";
}
也就是說,如果我要想定義一個可以延遲到一定實際再使用並初始化的 final 變數,這在 Java 中是做不到的。
Kotlin 有辦法,使用 lazy 這個 delegate 即可:
class Hello{
private val name by lazy{
NameProvider.getName()
}
}
只有使用到 name 這個屬性的時候,lazy 後面的 Lambda 才會執行,name 的值才會真正計算出來。
11. 如何獲得 class 的例項
Java 當中:
public class Hello{
...
}
...
Class<?> clazz = Hello.class;
Hello hello = new Hello();
Class<?> clazz2 = hello.getClass();
前面我們展示了兩種獲得 class 的途徑,一種直接用類名,一種通過類例項。剛剛接觸 Kotlin 的時候,獲取 Java Class 的方法卻是容易讓人困惑。
class Hello
val clazz = Hello::class.java
val hello = Hello()
val clazz2 = hello.javaClass
同樣效果的 Kotlin 程式碼看上去確實很奇怪,實際上 Hello::class 拿到的是 Kotlin 的 KClass,這個是 Kotlin 的型別,如果想要拿到 Java 的 Class 例項,那麼就需要前面的辦法了。
更多精彩內容歡迎關注騰訊 Bugly的微信公眾賬號:
騰訊 Bugly是一款專為移動開發者打造的質量監控工具,幫助開發者快速,便捷的定位線上應用崩潰的情況以及解決方案。智慧合併功能幫助開發同學把每天上報的數千條 Crash 根據根因合併分類,每日日報會列出影響使用者數最多的崩潰,精準定位功能幫助開發同學定位到出問題的程式碼行,實時上報可以在釋出後快速的瞭解應用的質量情況,適配最新的 iOS, Android 官方作業系統,鵝廠的工程師都在使用,快來加入我們吧!