一、继承

1.1 Scala中的继承结构

Scala 中继承关系如下图:

  • Any 是整个继承关系的根节点;
  • AnyRef 包含 Scala Classes 和 Java Classes,等价于 Java 中的 java.lang.Object;
  • AnyVal 是所有值类型的一个标记;
  • Null 是所有引用类型的子类型,唯一实例是 null,可以将 null 赋值给除了值类型外的所有类型的变量;
  • Nothing 是所有类型的子类型。

1.2 extends & override

Scala 的集成机制和 Java 有很多相似之处,比如都使用 extends 关键字表示继承,都使用 override 关键字表示重写父类的方法或成员变量。示例如下:

//父类
class Person {var name = ""// 1.不加任何修饰词,默认为 public,能被子类和外部访问var age = 0// 2.使用 protected 修饰的变量能子类访问,但是不能被外部访问protected var birthday = ""// 3.使用 private 修饰的变量不能被子类和外部访问private var sex = ""def setSex(sex: String): Unit = {this.sex = sex}// 4.重写父类的方法建议使用 override 关键字修饰override def toString: String = name + ":" + age + ":" + birthday + ":" + sex}

使用 extends 关键字实现继承:

// 1.使用 extends 关键字实现继承
class Employee extends Person {override def toString: String = "Employee~" + super.toString// 2.使用 public 或 protected 关键字修饰的变量能被子类访问def setBirthday(date: String): Unit = {birthday = date}}

测试继承:


object ScalaApp extends App {val employee = new Employeeemployee.name = "heibaiying"employee.age = 20employee.setBirthday("2019-03-05")employee.setSex("男")println(employee)
}// 输出: Employee~heibaiying:20:2019-03-05:男

1.3 调用超类构造器

在 Scala 的类中,每个辅助构造器都必须首先调用其他构造器或主构造器,这样就导致了子类的辅助构造器永远无法直接调用超类的构造器,只有主构造器才能调用超类的构造器。所以想要调用超类的构造器,代码示例如下:

class Employee(name:String,age:Int,salary:Double) extends Person(name:String,age:Int) {.....
}

1.4 类型检查和转换

想要实现类检查可以使用 isInstanceOf,判断一个实例是否来源于某个类或者其子类,如果是,则可以使用 asInstanceOf 进行强制类型转换。

object ScalaApp extends App {val employee = new Employeeval person = new Person// 1. 判断一个实例是否来源于某个类或者其子类 输出 true println(employee.isInstanceOf[Person])println(person.isInstanceOf[Person])// 2. 强制类型转换var p: Person = employee.asInstanceOf[Person]// 3. 判断一个实例是否来源于某个类 (而不是其子类)println(employee.getClass == classOf[Employee])}

1.5 构造顺序和提前定义

1. 构造顺序

在 Scala 中还有一个需要注意的问题,如果你在子类中重写父类的 val 变量,并且超类的构造器中使用了该变量,那么可能会产生不可预期的错误。下面给出一个示例:

// 父类
class Person {println("父类的默认构造器")val range: Int = 10val array: Array[Int] = new Array[Int](range)
}//子类
class Employee extends Person {println("子类的默认构造器")override val range = 2
}//测试
object ScalaApp extends App {val employee = new Employeeprintln(employee.array.mkString("(", ",", ")"))}

这里初始化 array 用到了变量 range,这里你会发现实际上 array 既不会被初始化 Array(10),也不会被初始化为 Array(2),实际的输出应该如下:

父类的默认构造器
子类的默认构造器
()

可以看到 array 被初始化为 Array(0),主要原因在于父类构造器的执行顺序先于子类构造器,这里给出实际的执行步骤:

  1. 父类的构造器被调用,执行 new Array[Int](range) 语句;
  2. 这里想要得到 range 的值,会去调用子类 range() 方法,因为 override val 重写变量的同时也重写了其 get 方法;
  3. 调用子类的 range() 方法,自然也是返回子类的 range 值,但是由于子类的构造器还没有执行,这也就意味着对 range 赋值的 range = 2 语句还没有被执行,所以自然返回 range 的默认值,也就是 0。

这里可能比较疑惑的是为什么 val range = 2 没有被执行,却能使用 range 变量,这里因为在虚拟机层面,是先对成员变量先分配存储空间并赋给默认值,之后才赋予给定的值。想要证明这一点其实也比较简单,代码如下:

class Person {// val range: Int = 10 正常代码 array 为 Array(10)val array: Array[Int] = new Array[Int](range)val range: Int = 10  //如果把变量的声明放在使用之后,此时数据 array 为 array(0)
}object Person {def main(args: Array[String]): Unit = {val person = new Personprintln(person.array.mkString("(", ",", ")"))}
}

2. 提前定义

想要解决上面的问题,有以下几种方法:

(1) . 将变量用 final 修饰,代表不允许被子类重写,即 final val range: Int = 10

(2) . 将变量使用 lazy 修饰,代表懒加载,即只有当你实际使用到 array 时候,才去进行初始化;

lazy val array: Array[Int] = new Array[Int](range)

(3) . 采用提前定义,代码如下,代表 range 的定义优先于超类构造器。

class Employee extends {//这里不能定义其他方法override val range = 2
} with Person {// 定义其他变量或者方法def pr(): Unit = {println("Employee")}
}

但是这种语法也有其限制:你只能在上面代码块中重写已有的变量,而不能定义新的变量和方法,定义新的变量和方法只能写在下面代码块中。

注意事项:类的继承和下文特质 (trait) 的继承都存在这个问题,也同样可以通过提前定义来解决。虽然如此,但还是建议合理设计以规避该类问题。

二、抽象类

Scala 中允许使用 abstract 定义抽象类,并且通过 extends 关键字继承它。

定义抽象类:

abstract class Person {// 1.定义字段var name: Stringval age: Int// 2.定义抽象方法def geDetail: String// 3. scala 的抽象类允许定义具体方法def print(): Unit = {println("抽象类中的默认方法")}
}

继承抽象类:

class Employee extends Person {// 覆盖抽象类中变量override var name: String = "employee"override val age: Int = 12// 覆盖抽象方法def geDetail: String = name + ":" + age
}

三、特质

3.1 trait & with

Scala 中没有 interface 这个关键字,想要实现类似的功能,可以使用特质 (trait)。trait 等价于 Java 8 中的接口,因为 trait 中既能定义抽象方法,也能定义具体方法,这和 Java 8 中的接口是类似的。

// 1.特质使用 trait 关键字修饰
trait Logger {// 2.定义抽象方法def log(msg: String)// 3.定义具体方法def logInfo(msg: String): Unit = {println("INFO:" + msg)}
}

想要使用特质,需要使用 extends 关键字,而不是 implements 关键字,如果想要添加多个特质,可以使用 with 关键字。

// 1.使用 extends 关键字,而不是 implements,如果想要添加多个特质,可以使用 with 关键字
class ConsoleLogger extends Logger with Serializable with Cloneable {// 2. 实现特质中的抽象方法def log(msg: String): Unit = {println("CONSOLE:" + msg)}
}

3.2 特质中的字段

和方法一样,特质中的字段可以是抽象的,也可以是具体的:

  • 如果是抽象字段,则混入特质的类需要重写覆盖该字段;
  • 如果是具体字段,则混入特质的类获得该字段,但是并非是通过继承关系得到,而是在编译时候,简单将该字段加入到子类。
trait Logger {// 抽象字段var LogLevel:String// 具体字段var LogType = "FILE"
}

覆盖抽象字段:

class InfoLogger extends Logger {// 覆盖抽象字段override var LogLevel: String = "INFO"
}

3.3 带有特质的对象

Scala 支持在类定义的时混入 父类 trait,而在类实例化为具体对象的时候指明其实际使用的 子类 trait。示例如下:

trait Logger:

// 父类
trait Logger {// 定义空方法 日志打印def log(msg: String) {}
}

trait ErrorLogger:

// 错误日志打印,继承自 Logger
trait ErrorLogger extends Logger {// 覆盖空方法override def log(msg: String): Unit = {println("Error:" + msg)}
}

trait InfoLogger:

// 通知日志打印,继承自 Logger
trait InfoLogger extends Logger {// 覆盖空方法override def log(msg: String): Unit = {println("INFO:" + msg)}
}

具体的使用类:

// 混入 trait Logger
class Person extends Logger {// 调用定义的抽象方法def printDetail(detail: String): Unit = {log(detail)}
}

这里通过 main 方法来测试:

object ScalaApp extends App {// 使用 with 指明需要具体使用的 trait  val person01 = new Person with InfoLoggerval person02 = new Person with ErrorLoggerval person03 = new  Person with InfoLogger with ErrorLoggerperson01.log("scala")  //输出 INFO:scalaperson02.log("scala")  //输出 Error:scalaperson03.log("scala")  //输出 Error:scala}

这里前面两个输出比较明显,因为只指明了一个具体的 trait,这里需要说明的是第三个输出,因为 trait 的调用是由右到左开始生效的,所以这里打印出 Error:scala

3.4 特质构造顺序

trait 有默认的无参构造器,但是不支持有参构造器。一个类混入多个特质后初始化顺序应该如下:

// 示例
class Employee extends Person with InfoLogger with ErrorLogger {...}
  1. 超类首先被构造,即 Person 的构造器首先被执行;
  2. 特质的构造器在超类构造器之前,在类构造器之后;特质由左到右被构造;每个特质中,父特质首先被构造;
    • Logger 构造器执行(Logger 是 InfoLogger 的父类);
    • InfoLogger 构造器执行;
    • ErrorLogger 构造器执行;
  3. 所有超类和特质构造完毕,子类才会被构造。

Scala 继承和特质相关推荐

  1. scala中使用特质中的抽象字段和实际字段

    Scala中,trait相当于Java中的接口,遇到需要使用Java接口的场景时,你就在scala中可以使用trait了. 我们知道Java中你可以实现多个接口,那么Scala中,你也可以继承多个tr ...

  2. scala入门-07特质类(trait)的使用

    trait类似于Java8中的可用带default method的接口. trait中可以带有实现的方法,也可以带有抽象方法,使用trait的方法是with而混入类中. 我们在scala下的org.s ...

  3. scala基础之特质trait

    Scala中,trait相当于Java中的接口,遇到需要使用Java接口的场景时,你就在Scala中可以使用trait了. 我们知道Java中你可以实现多个接口,那么Scala中,你也可以继承多个tr ...

  4. 四, Scala 伴生对象, 特质

    文章目录 四, Scala 伴生对象和伴生类 4.1 单例对象和伴生对象 4.1.1 什么是单例对象? 4.1.2 如何使用Scala的伴生对象和伴生类来实现单例模式? 4.1.2 apply方法 4 ...

  5. Scala 中的 特质(trait)

    文章目录 特质(trait) 概念 语法 继承特质 继承单个trait 代码示例 继承多个特质 代码示例 定义具体的方法 代码示例 trait中定义具体的字段和抽象的字段 定义 代码示例 模板模式 代 ...

  6. Scala特证/特质【6.7 特质(Trait)】

    Scala特证/特质[6.7 特质(Trait)] 6.7 特质(Trait) Java 的接口 接口的作用 抽象类的作用 6.7.1 特质声明 6.7.2 特质基本语法 6.7.3 特质叠加 6.7 ...

  7. scala之product特质理解

    最近在看sparksql源码的时候,发现TreeNode是继承scala的Product特质,所以比较好奇,就找了相关资料,通过编写demo,进行理解Product的具体用法. case class ...

  8. Scala学习之路 (六)Scala的类、对象、继承、特质

    一.类 1.类的定义 scala语言中没有static成员存在,但是scala允许以某种方式去使用static成员 这个就是伴生机制,所谓伴生,就是在语言层面上,把static成员和非static成员 ...

  9. scala面向对象基础---类继承和特质

    一.类继承 传送门:Scala基础-类继承 Scala的类继承 调用超类的构造方法 重写超类的成员 3.1.不被继承的成员 3.2.不可重写的成员 3.3.无参方法与字段 子类型多态与动态绑定 抽象类 ...

最新文章

  1. 600余名外出务工者免费乘高铁“返乡专列”回云南过春节
  2. 鱼缸式百分比信息图表,这样计算才正确
  3. 第十六届智能车竞赛创意组比赛-筹划初稿
  4. html css3d效果,html,css的3D变形
  5. 被前公司辞退后,前领导打电话命令你给前同事解释代码,该怎么办?
  6. js中关于new Object时传参的一些细节分析
  7. Comware、VRP、IOS这些操作系统平台你分清了吗?
  8. linux+qt+定时精度,Qt QTimer测试定时精度
  9. Vue——vue-resource
  10. android listview 增加单选 复选,ListView里面加入CheckBox如何实现单选?
  11. 【linux】nohup运行守护进程
  12. 用于 Windows8 的 Wijmo Charts 图表控件
  13. 初探12306售票算法(一)- 理论
  14. 我的世界服务器物品锁bug,【MOD教程】已知MOD服BUG物品解析
  15. 计算机无线网络计算机文件共享,无线局域网共享_在同一个无线局域网内如何共享文件?...
  16. matlab mat转bmp,mat格式转换
  17. 【蓝桥杯每日一练:蹩脚两轮车】
  18. 全球与中国胶原蛋白敷料市场深度研究分析报告
  19. 一个资源丰富的在线小程序社区推荐
  20. 运行tensorflow-datasets遇到import tensorflow.compat.v2 as tf报错ImportError: No module named tensorflow.V2

热门文章

  1. JAVA图片与字节流的相互转换
  2. int数组转strpython_python中int与str互转方法
  3. Mongo数据库的操作
  4. java实现qq邮箱发送附件和图片
  5. 比特大陆IPO之旅即将终结
  6. Newline required at end of file but not found.
  7. Google工具包Guava——聊聊代码校验Preconditions
  8. pdf2html java_pdf2HtmlEX的使用
  9. 转载:12个医学公共数据库
  10. SICK LMS511 LiDAR系统集成