Kotlin中的抽象类实现

抽象方法是只有方法签名,没有方法实现的方法。

有abstract修饰的成员,无须使用open修饰,当使用abstract修饰类时,表明这个类需要被继承;当使用abstract修饰方法、属性时,表明这个方法、属性必须由子类提供实现。

一、抽象成员和抽象类

抽象成员和抽象类必须使用abstract修饰符来定义,包含抽象成员的类智能被定义成抽象类,抽象类中可以没有抽象成员。

抽象方法和抽象类的规则如下:

  • 抽象类必须使用abstract修饰符来修饰,抽象爱你个成员也必须使用abstract修饰符来修饰,抽象方法不能有方法体。
  • 抽象类不能被实例化,无法调用抽象类的构造器创建抽象类的实例。即使抽象类中不包含任何抽象成员,这个抽象类也不能创建实例。
  • 抽象类可以包含属性、方法、构造器、初始化块、嵌套类5种成员。
  • 含有抽象成员的类智能被定义成抽象类。

定义抽象方法,只需在普通方法上增加abstract修饰符,并把普通方法的方法体全部去掉即可。

abstract class Shape {
  init {
    println("执行Shape的初始化块......")
  }

  var color = ""
  abstract fun calPerimeter(): Double

  abstract val type: String

  constructor() {}

  constructor(color: String) {
    println("执行Shape的构造器...")
    this.color = color
  }
}

抽象类不能用于创建实例,只能当作父类被其子类继承。

class Triangle(
  color: String, var a: Double,
  var b: Double, var c: Double
) : Shape(color) {
  fun setSides(a: Double, b: Double, c: Double) {
    if (a >= b + c || b >= a + c || c >= a + b) {
      println("三角形两边之和必须大于第三边")
      return
    }
    this.a = a
    this.b = b
    this.c = c
  }

  //重写Shape类的计算周长的抽象方法
  override fun calPerimeter(): Double {
    return a + b + c
  }

  //重写Shape类的代表形状的抽象属性
  override val type: String = "三角形"
}
class Circle(color: String, var radius: Double) : Shape(color) {
  override fun calPerimeter(): Double = 2 * Math.PI * radius

  override val type: String = "圆形"
}

fun main(args: Array<String>) {
  var s1: Shape = Triangle("黑色", 3.0, 4.0, 5.5)
  var s2: Shape = Circle("黄色", 4.0)
  println(s1.type)
  println(s2.type)

  println(s1.calPerimeter())
  println(s2.calPerimeter())
}

输出结果:

执行Shape的初始化块......
执行Shape的构造器...
执行Shape的初始化块......
执行Shape的构造器...
三角形
圆形
12.5
25.132741228718345

利用抽象类和抽象方法的优势,可以更好地发挥多态的优势,使得程序更加灵活。

注意:

  • abstract不能用于修饰局部变量,Kotlin中没有抽象变量的说法;
  • abstract也不能用于修饰构造器,没有抽象构造器,抽象类中定义的构造器只能是普通构造器。
  • 使用abstract关键字修饰的方法必须被其子类重写才有意义;
  • private和abstract不能同时修饰方法。

二、抽象类的作用

抽象类体现的就是一种模板模式的设计,抽象类作为多个子类的通用模板,子类在抽象类的基础上进行扩展、改造,但子类总体上会大致保留抽象类的行为方式。

//定义带转速属性的主构造器
abstract class SpeedMeter(var turnRate: Double) {
  //把返回车轮半径的方法定义成抽象方法
  abstract fun calGirth(): Double

  //定义计算速度的通用算法
  fun getSpeed(): Double {
    //速度等于车轮周长*转速
    return calGirth() * turnRate
  }
}

public class CarSpeedMeter(var radius: Double) : SpeedMeter(0.0) {
  override fun calGirth(): Double {
    return radius * 2 * Math.PI
  }
}

fun main(args: Array<String>) {
  val csm = CarSpeedMeter(0.28)
  csm.turnRate = 15.0
  println(csm.getSpeed())
}

输出结果:

26.389378290154266

下面是模板模式的一些简单规则:

  • 抽象父类可以只定义需要使用的某些方法,把不能实现的部分抽象成抽象方法,留给其子类去实现。
  • 父类中可能包含需要调用其他系列方法的方法,这些被调方法既可以由父类实现,也可以由其子类实现。

三、密封类

密封类是一种特殊的抽象类,转么用于派生子类。

密封类与普通抽象类的区别在于:密封类的子类是固定的。密封类的子类必须与密封类本身在同一个文件中,在其他文件中则不能为密封类派生子类。

//定义一个密封类
sealed class Apple {
  abstract fun taste()
}

open class RedFuji : Apple() {
  override fun taste() {
    println("红富士苹果今年真贵,但是还是很甜。")
  }
}

data class Gala(var weight: Double) : Apple() {
  override fun taste() {
    println("嘎啦苹果也不便宜,但更清脆,重量为${weight}")
  }
}

fun main(args: Array<String>) {
  var ap1: Apple = RedFuji()
  var ap2: Apple = Gala(3.5)
  ap1.taste()
  ap2.taste()
}

输出结果:

红富士苹果今年真贵,但是还是很甜。
嘎啦苹果也不便宜,但更清脆,重量为3.5

  • 密封类的本质就是抽象类。
  • 密封类的所有构造器都必须是private的,无论是否使用private修饰,系统都会自动添加private修饰。
  • 密封类的直接子类必须与密封类位于同一个文件中,但密封类的间接子类则无需在同一个文件中。

使用密封类的好处:

密封类的子类是固定的,可以清楚地知道密封类只可能有固定数量的子类。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Kotlin实现静态方法

    工具类 全都是静态方法的情况 : class 类名 改为 object 类名 即可 package redwolf.com.moreimageupload import okhttp3.MultipartBody import java.io.File /** * @作者 RedWolf * @时间 2017/5/20 10:52 * @简介 MoreImageUtils.kt */ object MoreImageUtils { fun filesToMultipartBodyParts(fi

  • Kotlin实现多函数接口的简化调用

    对于一个声明了多个方法的接口,我们使用的时候有时仅仅关注于几个关键方法,并不需要实现所有的.可是由于接口调用的语法限制,使得我们不得不在代码中也显示声明实现了那些我们不关心的方法.在Java中也有简化接口调用的方式,比如安卓中ViewPager监听页面切换时的接口PageChangeListener,官方提供了简单类: ViewPager.SimpleOnPageChangeListener来简化调用. 对于Kotlin来说,可以类似使用java的方式,来实现多函数接口的简化调用,只是要用到ob

  • 详解Kotlin:forEach也能break和continue

    详解Kotlin:forEach也能break和continue 这样的问题.也就是说,他们想用forEach而不是for循环,因为这很fp,很洋气(我也喜欢), 但是他们又想使用break和continue,也就是普通的流程控制语句中的控制语句. 这很不fp,因为原本有filter是用于完成这个工作的,还有flapMap.BennyHuo在他发的文章里面也说的是这种方法. filter很fp,但是会导致两次遍历,这样的话给人一股效率很低的赶脚.而Java8的Stream API就只会遍历一次,

  • kotlin 官方学习教程之基础语法详解

    kotlin 官方学习教程之基础语法详解 Google 在今天的举行了 I/O 大会,大会主要主要展示内有容 Android O(Android 8.0)系统.Google Assistant 语音助手.Google 智能音箱.人工智能.机器学习.虚拟现实等.作为一个 Android 开发者,我关心的当然是 Android O(Android 8.0)系统了,那么关于 Android O 系统的一个重要消息是全面支持 Kotlin 编程语言,使得 Kotlin 成为了 Android 开发的官方

  • kotlin和Java的相互调用示例详解

    前言 互操作就是在Kotlin中可以调用其他编程语言的接口,只要它们开放了接口,Kotlin就可以调用其成员属性和成员方法,这是其他编程语言所无法比拟的.同时,在进行Java编程时也可以调用Kotlin中的API接口. 1.在kotlin中调用Java方法 Kotlin和Java是两种不同的语言,所以在互相调用的时候,会有一些特殊的语法.kotlin中对象属性默认就带有setter和getter方法,所以在kotlin中调用Java时直接变量名点属性就可获取到属性的setter和getter的一

  • Kotlin编写Android适配器Adapter

    说好今天要写一个使用Kotlin写Adapter的列子,我想了半天也没有组织好语言,直接上代码吧,有一定Android基础的小伙伴肯定是能看的懂的 package com.example.administrator.kotlintest import android.content.Context import android.view.LayoutInflater import android.view.View import android.view.ViewGroup import and

  • 详解Kotlin中如何实现类似Java或C#中的静态方法

    大家可以在网络上搜到不少这样的文章,官方推荐是包级函数,也有人说用伴生对象(companion class).这些都是不错的选择,但并不完善,我们在不同的情况下有更好的选择.我总结了几种方法,分别是:包级函数.伴生对象.扩展函数和对象声明.这需要大家根据不同的情况进行选择. 一.包级函数 Kotlin和Java及C#不同的是,可以在包里面直接声明函数.做法和类中是一样的,这里就不多说了,的确是一个非常好的选择.适用于函数不需要不包内部的类进行数据共享的方法. 二.伴生对象 从语义上来讲,伴生函数

  • Kotlin 基础教程之数组容器

    Kotlin 基础教程之数组容器 Arrays Kotlin 标准库提供了arrayOf()创建数组, **ArrayOf创建特定类型数组 val array = arrayOf(1, 2, 3) val countries = arrayOf("UK", "Germany", "Italy") val numbers = intArrayOf(10, 20, 30) val array1 = Array(10, { k -> k * k

  • Kotlin基础学习之位运算

    什么是位运算? 程序中的所有数在计算机内存中都是以二进制的形式储存的.位运算说穿了,就是直接对整数在内存中的二进制位进行操作.比如,and运算本来是一个逻辑运算符,但整数与整数之间也可以进行and运算.举个例子,6的二进制是110,11的二进制是1011,那么6 and 11的结果就是2,它是二进制对应位进行逻辑运算的结果(0表示False,1表示True,空位都当0处理): 110 AND 1011 ---- 0010 –> 2 由于位运算直接对内存数据进行操作,不需要转成十进制,因此处理速度

  • 用Kotlin实现Android点击事件的方法

    近期,Google宣布Kotlin成为了Android一级开发语言.于是就刚刚简单的研究了一下,查资料的时候发现现成的资料还是很少的,于是决定自己记录一下,方便以后查看,也供其他人一个参考. 在android中,点击事件大致分为三种写法: 1. 匿名内部类. 2. Activity实现全局OnClickListener接口. 3. 指定xml的onClick属性. 今天用Kotlin实现这三种方式实现点击事件 匿名内部类:这种方式最简单 override fun onCreate(savedIn

随机推荐