Kotlin 中类的继承与方法重载

在 Kotlin 中,‌类的继承 ‌和‌方法重载 ‌需要明确使用 open 关键字,因为 Kotlin 默认所有类和方法都是 final(不可继承/不可重写)。以下是详细规则和示例:


一、类的继承

  1. 默认不可继承

    Kotlin 的类默认是 final,无法被继承。需用 open 标记类才能允许继承。

    kotlin 复制代码
    open class Animal(val name: String) // 允许被继承
    class Dog(name: String) : Animal(name) // 正确
    c 复制代码
    kotlinCopy Code
    class Plant // 默认 final,无法被继承
    // class Flower : Plant()  // 编译错误:Plant 是 final 的
  2. 继承语法

    子类必须调用父类的构造器(主构造器或次构造器)。

    kotlin 复制代码
    open class Parent
    class Child : Parent() // ✅ 正确(调用无参构造器)

二、方法重载

  1. 默认不可重写

    方法默认是 final,需用 open 标记允许子类重写。

    kotlin 复制代码
    open class Animal {
        open fun speak() { // 允许被重写
            println("...")
        }
    }
    
    class Dog : Animal() {
        override fun speak() { // 必须用 override
            println("Woof!")
        }
    }
  2. 禁止重写

    子类中可用 final 禁止进一步重写:

    kotlin 复制代码
    class Cat : Animal() {
        final override fun speak() { 
            println("Meow!")
        }
    }
    
    class Kitten : Cat() {
        // override fun speak() // 编译错误:speak 是 final 的
    }

类与方法重载案例

kotlin 复制代码
 open class Person(private val name:String){

     private fun showName() = "父类的姓名是:$name"

     open fun printlnName() = println("$name")
}

class Student(private val subName:String):Person(subName){

    private fun showName() = "子类 的姓名是【${subName}】"


    override fun printlnName() = println(showName())
}

fun main(){
    val student = Student("张三")
    student.printlnName()
}

三、属性的重写

  1. ‌**open 属性**‌

    属性(val/var)也可用 open 允许重写:

    kotlin 复制代码
    open class Shape {
        open val vertices: Int = 0
    }
    
    class Triangle : Shape() {
        override val vertices: Int = 3
    }
  2. ‌**val 重写为 var**‌

    可以将父类的 val 重写为子类的 var

    kotlin 复制代码
    open class Base {
        open val value: Int = 0
    }
    
    class Derived : Base() {
        override var value: Int = 1 // ✅ 允许
    }

四、多层继承中的 open

  • 父类的方法被重写后,默认仍是 open 的,可用 final 关闭:

    kotlin 复制代码
    open class A {
        open fun foo() {}
    }
    
    open class B : A() {
        final override fun foo() {} // 禁止 B 的子类重写 foo
    }
    
    class C : B() {
        // override fun foo() // 编译错误:foo 是 final 的
    }

五、常见错误

  1. ‌**忘记 open**‌

    kotlin 复制代码
    class Parent { // 默认 final
        fun method() {} // 默认 final
    }
    
    class Child : Parent() { // ❌ 编译错误:Parent 不可继承
        override fun method() {} // ❌ 编译错误:method 不可重写
    }
  2. ‌**忘记 override**‌

    kotlin 复制代码
    open class Parent {
        open fun method() {}
    }
    
    class Child : Parent() {
        fun method() {} // ❌ 编译错误:缺少 override
    }

六、与 abstract 的区别

  • abstract 类默认是 open 的,无需显式标记:

    kotlin 复制代码
    abstract class AbstractBase {
        abstract fun mustOverride() // 必须被重写
    }

总结

  • ‌**open**‌:标记类/方法/属性允许被继承或重写。
  • ‌**override**‌:强制要求显式标记重写的成员。
  • ‌**final**‌:在子类中禁止进一步重写。
相关推荐
时光少年6 分钟前
Android ExoPlayer版本升级遇上系统的”瓜“
android·前端
你说你说你来说2 小时前
安卓布局详解
android·笔记
奔跑吧 android2 小时前
【android bluetooth 框架分析 02】【Module详解 3】【HciHal 模块介绍】
android·bluetooth·bt·gd·aosp13·hcihal
好学人2 小时前
Activity的四种启动模型
android
好学人3 小时前
一文了解 Android MVI 架构
android
增强3 小时前
Jetpack Compose + CameraX+ MlKit 实现 二维码扫描(二)
android
studyForMokey3 小时前
【Android读书笔记】读书笔记记录
android
包包打太极4 小时前
android Jetpack Compose项目的build.gradle.kts,Gradle构建系统中的配置文件,用于定义Android项目的构建逻辑依赖
android
顾林海4 小时前
Flutter 文本组件深度剖析:从基础到高级应用
android·前端·flutter
追随远方5 小时前
Android Cmake构建的项目,需不需要配置指定ndk及版本
android·音频编解码