一、面向对象的概念

  • 本质上就是解决如何用程序描述世界的问题
  • 讨论如何把实际存在的东西映射成程序的类和对象
  • 一种程序设计的思路,思想,方法
  • 程序设计层面的概念
  • 设计模式:前人的程序设计经验

二、抽象类与接口

相当于半成品与协议

1、接口

  • 接口,直观理解就是一种约定
interface InputDevice{
    fun input(event: Any)
}
  • 接口不能有状态
  • 必须由类对其进行实现后使用

2、抽象类

  • 实现了一部分协议的半成品
  • 可以有状态,可以有方法实现
  • 必须由子类继承后使用

3、抽象类和接口的共性

  • 比较抽象,不能直接实例化
  • 需要有子类(实现类)实现的方法
  • 父类(接口)变量可以接受子类(实现类)的赋值

3、抽象类和接口的区别

  • 抽象类有状态,接口没有状态
  • 抽象类有方法实现,接口只能有无状态的默认实现
  • 抽象类只能单继承,接口可以多实现
  • 抽象类反应本质,接口体现能力

三、类的继承与接口的实现

  • 父类需要 open 才可以被继承
  • 父类方法、属性需要 open 才可以倍复写
  • 接口、接口方法、抽象类默认为 open
  • 复写父类(接口)成员需要 override 关键字
  • class D:A(),B,C
  • 注意继承类时实际上调用了父类构造方法
  • 类只能单继承,接口可以多实现

接口代理

  • class Manager(driver: Driver):Driver by driver
  • 接口方法实现交给代理类实现

接口方法冲突

- 接口方法可以有默认实现
- 签名一致且返回值相同的冲突
- 子类(实现类)必须复写冲突方法
- super<[父类(接口)名]>.[方法名]([参数列表])

四、类及其成员的可见性

Kotlin Java
private private
protected protected
default(包内可见)
internal(模块内可见)
public public

五、object

  • 只有一个实例的类(单例)
  • 不能自定义构造方法
  • 可以实现接口、继承父类
  • 本质上就是单例模式最基本的实现

六、伴生对象与静态成员

  • 每个类可以对应一个伴生对象
  • 伴生对象的成员全局独一份
  • 伴生对象的成员类似Java的静态成员
  • 静态成员考虑用包级函数、变量代替
  • 使用JvmField和JvmStatic,可以让伴生对象在Java中使用时看起来像静态成员和静态方法

七、方法重载(Overloads)和默认参数

重载
  • 写了一个方法和已经存在的方法重名,参数不相同
  • Jvm函数签名的概念:函数名、参数列表
  • 跟返回值没有关系
默认参数
  • 为函数参数设定一个默认值
  • 可以为任意位置的参数设置默认值
  • 函数调用产生混淆时用具名参数
方法重载与默认参数
  • 方法重载与默认参数可以互相转换,在Java中调用有默认参数的方法时,该方法需要使用注解@JvmOverloads
  • 避免定义关系不大的重载方法

八、扩展成员(二次加工)

  • 为现有类添加方法、属性

  • fun X.y():Z{...}
    val X.m
    //注意扩展属性不能初始化,类似接口属性
  • Java 调用扩展成员类似调用静态方法

九、属性代理

  • 定义方法

  • val/var <property name>: <Type> by <expression>
  • 代理者需要实现相应的setValue/getValue方法

  • by lazy 就是在第一次使用的时候才去实例化对象

十、数据类(data class)

  • 默认实现 copy()、toString()、equals()等方法
  • 编译器默认生成component[1-N]方法,可以自己复写
data class China(val code:Int,name: String)
val (a,b) = China(0,"中国")
pringln("$a -> $b")
  • allOpen和noArg 插件解决data class 的类被解析成final类,并且无空构造方法的问题
//build.gradle 中的 dependencies 中添加
dependencies{
    classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version"
    classpath "org.jetbrains.kotlin:kotlin-noarg:$kotlin_version"
    classpath "org.jetbrains.kotlin:kotlin-allopen:$kotlin_version"
}

//build.gradle 中 添加插件的应用
apply plugin: 'kotlin-noarg'
apply plugin: 'kotlin-allopen'

// 插件的配置

noArg{
    annotation("注解类名")
}

allOpen{
    annotation("注解类名")
}
  • 属性需要写到构造方法,初始化时需要传入参数

十一、内部类

  • 定义在类内部的类
  • 与类成员有相似的访问控制
  • 默认是静态内部类,非静态用 inner 关键字
  • this@Outter 与 this@Inner
匿名内部类
  • 没有定义名字的类
  • 类名编译时生成,类似Outter$1.class
  • 可继承父类、实现多个接口,与Java 不一样

十二、枚举类

  • 实例可数的类,注意枚举也是类
  • 可以修改构造方法,添加成员
  • 可以提升代码的表现力,也有一定的性能开销

十二、密封类(sealed class)

  • 子类可数的类
  • <V1.1,子类必须定义为密封类的内部类
  • v1.1开始,子类只需要与密封类在同一个文件中
  • 子类可以有多个构造参数
  • 表示状态适合使用枚举,需要传输指令时适合使用密封类