每天学一点 Kotlin -- 对象进阶:对象类型

----《第一季Kotlin崛起:次世代Android开发 》学习笔记

总目录:每天学一点 Kotlin ---- 目录
上一篇:每天学一点 Kotlin -- 多彩的类:委托1
下一篇:每天学一点 Kotlin -- 对象进阶:异常处理

1. 类型检查与类型转换

1.1 在 Kotlin 中,关键字 is 用来判断传入的类型是否是当前类型或者是它的子类型。这个对应于 Java 中的 instanceof 关键字。

1.2 举个栗子:

open class TypePerson(name: String, age: Int) {
    val name = name
    val age = age
    fun printPersonInfo() {
        println("printPersonInfo() -- name = $name, age = $age")
    }
}

class TypeStudent(name: String, age: Int, grade: Double) : TypePerson(name, age) {
    val grade = grade
    fun printStudentInfo() {
        println("printStudentInfo() -- name = $name, age = $age, grade = $grade")
    }
}

class TypeWorker(name: String, age: Int, salary: Double) : TypePerson(name, age) {
    val salary = salary
    fun printWorkerInfo() {
        println("printWorkerInfo() -- name = $name, age = $age, grade = $salary")
    }
}

fun testType1(p: TypePerson) {
    if (p is TypePerson) {
        p.printPersonInfo()
    } else if (p is TypeStudent) {
        p.printStudentInfo()
    } else if (p is TypeWorker) {
        p.printWorkerInfo()
    }
}

fun main() {
    var p1 = TypePerson("person", 20)
    var p2 = TypeStudent("student", 20, 80.8)
    var p3 = TypeWorker("worker", 30, 6000.0)

    testType1(p1)
    testType1(p2)
    testType1(p3)
}

打印结果:

printPersonInfo() -- name = person, age = 20
printPersonInfo() -- name = student, age = 20
printPersonInfo() -- name = worker, age = 30

和预想的结果不同,打印结果里面都是父类的方法打印。这是因为这里有一个陷阱,子类实例化后,不但是子类的类型,也是父类的类型。所以方法里面一直在走第一个 if 判断的逻辑。代码修改如下:

fun testType2(p: TypePerson) {
    if (p is TypeWorker) {
        p.printWorkerInfo()
    } else if (p is TypeStudent) {
        p.printStudentInfo()
    } else if (p is TypePerson) {
        p.printPersonInfo()
    }
}

fun main() {
    var p1 = TypePerson("person", 20)
    var p2 = TypeStudent("student", 20, 80.8)
    var p3 = TypeWorker("worker", 30, 6000.0)

//    testType1(p1)
//    testType1(p2)
//    testType1(p3)
    testType2(p1)
    testType2(p2)
    testType2(p3)
}

打印结果:

printPersonInfo() -- name = person, age = 20
printStudentInfo() -- name = student, age = 20, grade = 80.8
printWorkerInfo() -- name = worker, age = 30, grade = 6000.0

1.3 编译器对父类和子类之间是可以智能转换的,比如说一个方法中参数是父类,实际传入的是子类,这样是不会报错的,并且调用的方法也会是子类中的方法。

1.4 智能转换时是有规则的:
(1) 对于 val 局部常量总是可以的;
(2) 对于 val 属性,如果属性是 private 或 internal,或者该检查在声明属性的同一模块执行也是可以的,智能转换不适用于 open 的属性或者具有自定义 getter 的属性;
(3) var 局部变量,如果变量在检查和使用之间没有修改,并且没有在会修改它的 lambda 中捕获,也是可以的。
(4) 强调一点:var 属性是绝对不可能被智能转换的,因为这个变量可以随时被其他代码修改。

2. 对象的强制类型转换: as

2.1 也可以手动地进行类型转换。如下列代码中所示:

fun testType3(){
    var w : TypePerson = TypeWorker("worker", 45, 300.0)
    val w1 = w as TypeWorker
    w1.printWorkerInfo()
}

fun main() {
    testType3()
}

打印结果:

printWorkerInfo() -- name = worker, age = 45, grade = 300.0

在代码中,虽然是用子类进行实例化,但是声明的却是父类。因此变量 w 是父类类型的。然后在给 w1 赋值时,使用了关键字 as 将变量 w 强制转换为子类类型。之后调用了子类的方法并成功运行。

2.2 再举个栗子:不是父子类之间的转换

fun testType4() {
    val s: String = ""
    val s1 = s as TypeStudent
    s1.printStudentInfo() //  java.lang.ClassCastException: class java.lang.String cannot be cast to class chapter07.TypeStudent
}

fun main() {
    testType4()
}

打印结果:

java.lang.ClassCastException: class java.lang.String cannot be cast to class TypeStudent

在上面的代码中,运行的时候会报错,因为 String 类型无法转换成 TypeStudent 类型。

2.3 再举个栗子:可空类型之间的转换
(1) 举个栗子1:

fun testType5() {
    val w: TypeWorker? = TypeWorker("xiaoMing", 30, 400.0)
    val w1 = w as TypeWorker
    w1.printWorkerInfo()
}

fun main() { 
    testType5()
}

打印结果:

printWorkerInfo() -- name = xiaoMing, age = 30, grade = 400.0

(2) 举个栗子2:

fun testType6() {
    val w: TypeWorker? = null
    val w1 = w as TypeWorker
    w1.printWorkerInfo() // java.lang.NullPointerException: null cannot be cast to non-null type chapter07.TypeWorker
}

fun main() { 
    testType6()
}

打印结果:

 java.lang.NullPointerException: null cannot be cast to non-null type chapter07.TypeWorker

因为 TypeWorker? 类型是可以为 null 的,而 TypeWorker 类型不能为 null。由(1)和(2)对比可知:(1)中的变量 w 不为 null,转换是没有问题的;但是在(2)中的变量 w 是 null,因此转换时出错了。

2.4 在使用 as 进行强制类型转换时要注意可空类型, 所以不能保证一个变量不是空的,那么使用 as 反而不方便,不过,可以使用 as? 进行转换。举个栗子:

fun testType7() {
    val w: TypeWorker? = null
    val w1: TypeWorker? = w
    w1?.printWorkerInfo()
}

fun main() {
    testType7()
}

打印结果:
打印是空白的。

3. 总结

3.1 随着类的增加和代码量的提升,类型检查和类型转换显得越来越重要,在项目中使用要特别注意呢。

相关代码:https://gitee.com/fzq.com/test-demo
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,793评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,567评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,342评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,825评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,814评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,680评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,033评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,687评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,175评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,668评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,775评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,419评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,020评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,978评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,206评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,092评论 2 351
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,510评论 2 343

推荐阅读更多精彩内容