Scala 学习之路(九)—— 继承和特质
一、继承
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 Employee
employee.name = "heibaiying"
employee.age = 20
employee.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 Employee
val 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 = 10
val array: Array[Int] = new Array[Int](range)
}
//子类
class Employee extends Person {
println("子类的默认构造器")
override val range = 2
}
//测试
object ScalaApp extends App {
val employee = new Employee
println(employee.array.mkString("(", ",", ")"))
}
这里初始化array用到了变量range,这里你会发现实际上array既不会被初始化Array(10),也不会被初始化为Array(2),实际的输出应该如下:
父类的默认构造器
子类的默认构造器
()
可以看到array被初始化为Array(0),主要原因在于父类构造器的执行顺序先于子类构造器,这里给出实际的执行步骤:
- 父类的构造器被调用,执行
new Array[Int](range)
语句; - 这里想要得到range的值,会去调用子类range()方法,因为
override val
重写变量的同时也重写了其get方法; - 调用子类的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 Person
println(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: String
val 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 InfoLogger
val person02 = new Person with ErrorLogger
val person03 = new Person with InfoLogger with ErrorLogger
person01.log("scala") //输出 INFO:scala
person02.log("scala") //输出 Error:scala
person03.log("scala") //输出 Error:scala
}
这里前面两个输出比较明显,因为只指明了一个具体的trait
,这里需要说明的是第三个输出,因为trait的调用是由右到左开始生效的,所以这里打印出Error:scala
。
3.4 特质构造顺序
trait
有默认的无参构造器,但是不支持有参构造器。一个类混入多个特质后初始化顺序应该如下:
// 示例
class Employee extends Person with InfoLogger with ErrorLogger {...}
- 超类首先被构造,即Person的构造器首先被执行;
- 特质的构造器在超类构造器之前,在类构造器之后;特质由左到右被构造;每个特质中,父特质首先被构造;
- Logger构造器执行(Logger是InfoLogger的父类);
- InfoLogger构造器执行;
- ErrorLogger构造器执行;
- 所有超类和特质构造完毕,子类才会被构造。
参考资料
- Martin Odersky . Scala编程(第3版)[M] . 电子工业出版社 . 2018-1-1
- 凯.S.霍斯特曼 . 快学Scala(第2版)[M] . 电子工业出版社 . 2017-7
更多大数据系列文章可以参见个人 GitHub 开源项目: 程序员大数据入门指南
Scala 学习之路(九)—— 继承和特质的更多相关文章
- Scala学习之路 (六)Scala的类、对象、继承、特质
一.类 1.类的定义 scala语言中没有static成员存在,但是scala允许以某种方式去使用static成员这个就是伴生机制,所谓伴生,就是在语言层面上,把static成员和非static成员用 ...
- scala 学习笔记十二 继承
1.介绍 继承是面向对象的概念,用于代码的可重用性.可以通过使用extends关键字来实现继承. 为了实现继承,一个类必须扩展到其他类,被扩展类称为超类或父类.扩展的类称为派生类或子类. Scala支 ...
- Scala学习之路 (九)Scala的上界和下届
一.泛型 1.泛型的介绍 泛型用于指定方法或类可以接受任意类型参数,参数在实际使用时才被确定,泛型可以有效地增强程序的适用性,使用泛型可以使得类或方法具有更强的通用性.泛型的典型应用场景是集合及集合中 ...
- scala学习之路一
所谓学习,那么首先就先简单介绍一下scala吧 1.scala的介绍 Scala 是一门多范式(multi-paradigm)的编程语言,设计初衷是要集成面向对象编程和函数式编程的各种特性. Scal ...
- Scala 学习之路(十三)—— 隐式转换和隐式参数
一.隐式转换 1.1 使用隐式转换 隐式转换指的是以implicit关键字声明带有单个参数的转换函数,它将值从一种类型转换为另一种类型,以便使用之前类型所没有的功能.示例如下: // 普通人 clas ...
- Scala 学习之路(十二)—— 类型参数
一.泛型 Scala支持类型参数化,使得我们能够编写泛型程序. 1.1 泛型类 Java中使用<>符号来包含定义的类型参数,Scala则使用[]. class Pair[T, S](val ...
- Scala 学习之路(五)—— 集合类型综述
一.集合简介 Scala中拥有多种集合类型,主要分为可变的和不可变的集合两大类: 可变集合: 可以被修改.即可以更改,添加,删除集合中的元素: 不可变集合类:不能被修改.对集合执行更改,添加或删除操作 ...
- [原创]java WEB学习笔记87:Hibernate学习之路-- -映射 继承关系(subclass , joined-subclass,union-subclass )
本博客的目的:①总结自己的学习过程,相当于学习笔记 ②将自己的经验分享给大家,相互学习,互相交流,不可商用 内容难免出现问题,欢迎指正,交流,探讨,可以留言,也可以通过以下方式联系. 本人互联网技术爱 ...
- Scala学习之路 (八)Scala的隐式转换和隐式参数
一.概念 Scala 2.10引入了一种叫做隐式类的新特性.隐式类指的是用implicit关键字修饰的类.在对应的作用域内,带有这个关键字的类的主构造函数可用于隐式转换. 隐式转换和隐式参数是Scal ...
随机推荐
- 分位数(quantiles)、Z-score 与 F-score
0. 分位数(quantiles) 因为累计分布函数(cdf,F−1)是单调增函数,因此其有反函数,不妨记为 F−1. 其真实的含义在于,如果 F 是 X 的 cdf,则 F−1(α) 的函数值为: ...
- 计算机程序设计的史诗TAOCP
倘若你去问一个木匠学徒:你需要什么样的工具进行工作,他可能会回答你:“我只要一把锤子和一个锯”.但是如果你去问一个老木工或者是大师级的建筑师,他会告诉你“我需要一些精确的工具”.由于计算机所解决的问题 ...
- EF 导航属性的使用
using System;using System.Collections.Generic;using System.Linq;using System.Text;using System.Threa ...
- GDI+ Bitmap与WPF BitmapImage的相互转换
原文:GDI+ Bitmap与WPF BitmapImage的相互转换 using System.Windows.Interop; //... // Convert BitmapImage to Bi ...
- gcc/g++编译(生动形象,从最容易入手的hello world解释了库的概念)
1. gcc/g++在执行编译工作的时候,总共需要4步 (1).预处理,生成.i的文件[预处理器cpp] (2).将预处理后的文件不转换成汇编语言,生成文件.s[编译器egcs] (3).有汇编变为目 ...
- 图像滤镜艺术---Oilpaint油画滤镜
原文:图像滤镜艺术---Oilpaint油画滤镜 Oilpaint油画滤镜 图像油画效果实际上是将图像边缘产生一种朦胧,雾化的效果,同时,将一定的边缘模糊化,这样图像整体上看去像素与像素之间 ...
- Win10《芒果TV》商店版更新v3.2.5:新增会员频道,修复多处细节问题,小年快乐
听因乐不凡,尽在芒果TV,湖南卫视大型音乐竞技节目<歌手>,每周六晚22:30在芒果TV与湖南卫视同步直播,1月20日周五晚七点半,2016-2017湖南卫视<小年夜春晚>会员 ...
- android Choose library dependency 搜索不到目标库
问题:Choose library dependency 搜索不到目标库,百度了一下,发现尽是废话,无解,反正就是升级ide,我是 android studio是2.3.3(网上说升级到3.+就好了, ...
- 【redis】redis的bind配置
原文:[redis]redis的bind配置 在配置文件redis.conf中,默认的bind 接口是127.0.0.1,也就是本地回环地址.这样的话,访问redis服务只能通过本机的客户端连接, ...
- intellij开发安卓与genymotion配合
原文:intellij开发安卓与genymotion配合 [声明] 欢迎转载,但请保留文章原始出处→_→ 生命壹号:http://www.cnblogs.com/smyhvae/ 文章来源:http: ...