kotlin基础--对象、接口、抽象类

2021-12-06 17:19:01 浏览数 (1)

上次介绍了kotlin的类定义与初始化,接下来学习对象、接口、抽象类
一、对象
1.object关键字

object,类似Java中的静态

三种使用方式:

1.1 对象声明

对应Java中的单例类,只会在内存中实例化一次

代码语言:javascript复制
object Const {
    init {
        println("init")
    }

    fun getConfig(): String {
        return "config"
    }
}

fun main() {
    println(Const.getConfig())
    println(Const.getConfig())
}

结果:

init

config

config

1.2 对象表达式

Java中的匿名类

java写法如下:

代码语言:javascript复制
    static class Man {
        void doSomthing() {

        }
    }

    public static void main(String[] args) {
        Man my = new Man() {
            @Override
            void doSomthing() {
                super.doSomthing();
            }
        };
    }

kotlin写法:

代码语言:javascript复制
open class Man {
    open fun doSomthing() = ""
}

fun main() {
    var superMan = object : Man() {
        override fun doSomthing(): String {
            return "super Man protected earth"
        }
    }

    println(superMan.doSomthing())
}
1.3 伴生对象

如果你想要某个对象和一个类实例化绑定在一起,可以考虑伴生对象,使用 companion object 可以在一个类里定义一个伴生对象,一个类只能有一个伴生对象,

伴生对象也是静态的,只会在类实例化或调用伴生对象中的内容(对象和函数)时实例化一次

代码语言:javascript复制
class Const {
    init {
        println("init Const")
    }

    companion object {
        init {
            println("init companion object")
        }

        fun getConfig(): String {
            return "config"
        }
    }

}

fun main() {
    val const = Const()
    
    println(Const.getConfig())
    println(Const.getConfig())
}

结果:

init companion object

init Const

config

config

2.嵌套类

和Java中的内部类类似,如果一个类只对另一个类有用,那么使用内部类是合理的

代码语言:javascript复制
class Man {
    class BatMan {
        fun introduce() = "i'm Batman"
    }

    class IronMan {
        fun introduce() = "i'm Ironman"
    }
}

fun main() {
    println(Man.BatMan().introduce())
}
3.数据类

数据类是用来存储数据的类,它会自动实现toString、hashCode、equals的个性化实现

代码语言:javascript复制
class NormalClz(var x: Int, var y: Int) {

}

data class DataClz(var x: Int, var y: Int) {
    var z: Int = 40
}

fun main() {
    val normal = NormalClz(10, 10)
    println(normal)

    val data = DataClz(10, 10)
    println(data)

}

结果:

com.aruba.mykotlinapplication.NormalClz@5e481248

DataClz(x=10, y=10)

注意:数据类自动实现的个性化只对主构造函数里的定义的参数起作用

4.copy函数

数据类还提供了copy函数,用来复制一个对象

代码语言:javascript复制
data class DataClz(var x: Int, var y: Int) {
    var z: Int = 40

    override fun toString(): String {
        return "DataClz : $x $y $z"
    }

    constructor(_x: Int) : this(_x, 20) {
        //copy函数不会赋值
        this.z = 20
    }
}

fun main() {
    val data = DataClz(10, 10)
    println(data.copy(20))
}

结果:

DataClz : 20 10 40

注意:copy函数不会复制次构造函数中的赋值

5.解构声明

前面我们已经使用过解构语法了,如果想要在自己定义的类中使用,需要operate关键字,

并声明component1、component2...组件函数,函数名不能擅自改动

代码语言:javascript复制
class NormalClz(var x: Int, var y: Int) {
    operator fun component1() = x
    operator fun component2() = y
}

fun main() {
    val normal = NormalClz(20, 30)

    val (x, y) = normal
    println("$x $y")
}

数据类会自动为所有在主构造函数内定义的属性进行解构声明

代码语言:javascript复制
data class DataClz(var x: Int, var y: Int) {
}

fun main() {
    val data = DataClz(20, 30)

    val (x, y) = data
    println("$x $y")
}
6.使用数据类的条件
7.运算符重载

之前使用集合是我们可以直接使用 " " 、"-" 等来添加和删除元素

和c 一样,kotlin也支持运算符重载,只需要重载下面的函数就可以实现了:

8.枚举类

用来定义常量集合的一种特殊类

代码语言:javascript复制
enum class Position {
    east,
    west,
    south,
    north
}

也可以定义函数

代码语言:javascript复制
enum class Position(
    val data: DataClz
) {
    east(DataClz(1, 0)),
    west(DataClz(-1, 0)),
    south(DataClz(0, -1)),
    north(DataClz(0, 1));

    fun updatePosition(upData: DataClz) =
        DataClz(
            this.data.x   upData.x,
            upData.y   data.y
        )
}

fun main() {
    println(Position.east.data)
    println(Position.east.updatePosition(DataClz(10, 10)))
}
9.代数数据类型

可以用来表示一种子类型的闭集,枚举类就是一种简单的代数数据类型(ADT)

代码语言:javascript复制
enum class Position {
    east,
    west,
    south,
    north
}

class Treasure(var position: Position) {
    fun find() = when (position) {
        Position.east -> "不在东边"
        Position.west -> "不在西边"
        Position.south -> "不在南边"
        Position.north -> "你在北边找到了宝藏"
    }
}


fun main() {
    println(Treasure(Position.north).find())
}

结果:

你在北边找到了宝藏

10.密封类

对于更复杂的ADT,可以使用密封类,使用sealed修饰类,来实现更加复杂的定义,密封类可以更灵活的控制某个子类型

子类型必须和它定义在同一个文件里

代码语言:javascript复制
sealed class Position {
    //使用object,防止重复创建
    object east : Position()
    object west : Position()
    object south : Position()
    class north(var item: String) : Position()
}

class Treasure(var position: Position) {
    fun find() = when (position) {
        is Position.east -> "不在东边"
        is Position.west -> "不在西边"
        is Position.south -> "不在南边"
        is Position.north -> "你在北边找到了:${(position as Position.north).item}"
    }
}


fun main() {
    println(Treasure(Position.west).find())
    println(Treasure(Position.north("肥皂")).find())
}

结果:

不在西边

你在北边找到了:肥皂

二、接口

和Java差不多,用interface定义,实现接口的方法必须有override修饰符

代码语言:javascript复制
interface Callback {
    fun callback()
}

class Impl() : Callback {
    override fun callback() {
        TODO("not implemented") //To change body of created functions use File | Settings | File Templates.
    }
}

接口也可以定义函数实现和getter函数

代码语言:javascript复制
interface Callback {
    val number: Int
        get() = (0..500).shuffled().first()

    fun callback() = "haha"
}

class Impl() : Callback {
    override fun callback(): String {
        return super.callback()
    }
}
三、抽象类

和Java一样,可以用abstract修饰类和方法

代码语言:javascript复制
abstract class AbstractClass {
    abstract fun callback(): String
}

class ImplClz() : AbstractClass() {
    override fun callback(): String {
        TODO("not implemented") //To change body of created functions use File | Settings | File Templates.
    }
}

0 人点赞