一. 類(lèi)和接口
- 類(lèi)的定義
默認(rèn)為 public 可省略摩幔,類(lèi)內(nèi)無(wú)內(nèi)容可省略{}
class SimpleClass {
var x : Int = 0 // 必須初始化
constructor(x : Int){ // 構(gòu)造函數(shù)
this.x = x
}
}
可以簡(jiǎn)化為:
class SimpleClass ( var x : Int ){
}
- 類(lèi)的實(shí)例化
可以省掉new
val simpleClass = SimpleClass(9)
println(simpleClass.x)
simpleClass.y()
- 接口的定義
和Java沒(méi)有什么區(qū)別
interface SimpleInf {
fun simpleMethod()
}
接口的實(shí)現(xiàn)
class SimpleClass(var x : Int) : SimpleInf {
...
override fun simpleMethod(){
}
}
- 抽象類(lèi)的定義
和Java沒(méi)什么區(qū)別
abstract class AbsClass {
abstract fun absMethod()
open fun overridable(){}
fun nonOverridable(){}
}
需要注意:kotlin 中它所有的類(lèi)默認(rèn)都是 final 的,那么就意味著不能被繼承抖甘,而且在類(lèi)中所有的方法也默認(rèn)是 final 的热鞍,不能被重寫(xiě),如果想要繼承或重寫(xiě),只需要在類(lèi)或方法前添加 open 關(guān)鍵字即可薇宠。
- 屬性引用
val ageRef = Person::age // 未綁定receiver
val person = Person(18,"Jone")// 綁定receiver
val nameRef = person.name
ageRef.set(person,20)
nameRef.set("Andy")
二. 擴(kuò)展方法
- 定義
直接新建一個(gè) File 類(lèi)型的 kt 文件偷办,不是 class 類(lèi)型,不是 interface 類(lèi)型澄港,不是枚舉類(lèi)型椒涯,不是 Object 類(lèi)型,僅僅是 File 類(lèi)型回梧。
可以定義擴(kuò)展方法废岂、擴(kuò)展屬性。
// 類(lèi)型可以省略狱意,即任意類(lèi)型
fun 類(lèi)型.方法名(參數(shù)...):返回值
// 任意類(lèi)型都可以調(diào)用
fun add(a: Int, b: Int): Int{
return a + b
}
//只能 String 類(lèi)型可以調(diào)用
fun String.times(count: Int): String{
...
}
調(diào)用:
val a = 1
val b = 2
add(a, b)
val result: String
result.times(5)
三. 空類(lèi)型安全
- 可空類(lèi)型
!! 強(qiáng)制轉(zhuǎn)換為不空類(lèi)型
?. 安全調(diào)用成員
?: 前面的表達(dá)式為null 則執(zhí)行后面的表達(dá)式
var nullable: String? = "Hello"
val length = nullable!!.length // !! 強(qiáng)制轉(zhuǎn)成不空類(lèi)型湖苞,如果你確定不可能為空,不然還是會(huì)空指針
//或
val length = nullable?.length ?:0 // elvis 安全訪(fǎng)問(wèn)
- 空類(lèi)型的繼承關(guān)系
可空類(lèi)型為不可空類(lèi)型的父類(lèi)
var x: String = "Hello"
var y: String? = "World"
// x = y 详囤,不可以
// y = x财骨,可以
四. 智能類(lèi)型轉(zhuǎn)換
A as B 類(lèi)型轉(zhuǎn)換
A as? B 安全轉(zhuǎn)換,失敗返回 null
public interface Kotliner{}
public class Person implements Kotliner{
public final String name;
public final int age;
public Person(String name, int age) {
this.name = name;
this.age = age;
}
}
轉(zhuǎn)換:
val kotliner: Kotliner = Person("benny", 20)
if(kotliner is Person) {
//println((kotliner as Person).name) // kotliner 自動(dòng)轉(zhuǎn)換類(lèi)型為Person
println((kotliner as? Person)?.name) // 安全轉(zhuǎn)換
}
五. 建議
- 盡可能使用 val 來(lái)聲明不可變引用藏姐,讓程序的含義更加清晰確定隆箩;
- 盡可能減少函數(shù)對(duì)外部變量的訪(fǎng)問(wèn),也為函數(shù)式編程提供基礎(chǔ)羔杨;
- 必要時(shí)創(chuàng)建局部變量指向外部變量捌臊,避免因它變化引起程序錯(cuò)誤;
六. 案例
使用 Retrofit 發(fā)網(wǎng)絡(luò)請(qǐng)求
- 首先定義一個(gè)返回的數(shù)據(jù)結(jié)構(gòu)
data class Repository (
var id: Int,
var node_id: String,
var name: String,
var url: String
)
- 定義一個(gè)接口
interface GithubApi {
@GET("/repos/{owner}/{repo}")
fun getRepository(@Path("owner") owner: String, @Path("repo") repo: String): Call< Repository >
}
- 使用
val gitHubApi = Retrofit.Build()
.baseUrl("https://api.github.com")
.addConverterFactory(GsonConverterFactory.create())
.build()
.create(GithubApi::class.java)
val response = gitHubApi. getRepository("JetBrains","Kotlin").execute()
val repository = response.body()
if (repository == null) {
println("Error!${response.code()}-${response.message()}")
} else {
println(repository.name)
}