Scala 中继承关系以下图:java
Scala 的集成机制和 Java 有不少类似之处,好比都使用 extends
关键字表示继承,都使用 override
关键字表示重写父类的方法或成员变量。示例以下:git
//父类 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
关键字实现继承:github
// 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:男
在 Scala 的类中,每一个辅助构造器都必须首先调用其余构造器或主构造器,这样就致使了子类的辅助构造器永远没法直接调用超类的构造器,只有主构造器才能调用超类的构造器。因此想要调用超类的构造器,代码示例以下:ide
class Employee(name:String,age:Int,salary:Double) extends Person(name:String,age:Int) { ..... }
想要实现类检查可使用 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]) }
在 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),实际的输出应该以下:this
父类的默认构造器 子类的默认构造器 ()
能够看到 array 被初始化为 Array(0),主要缘由在于父类构造器的执行顺序先于子类构造器,这里给出实际的执行步骤:scala
new Array[Int](range)
语句;override val
重写变量的同时也重写了其 get 方法;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("(", ",", ")")) } }
想要解决上面的问题,有如下几种方法:
(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 }
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) } }
和方法同样,特质中的字段能够是抽象的,也能够是具体的:
trait Logger { // 抽象字段 var LogLevel:String // 具体字段 var LogType = "FILE" }
覆盖抽象字段:
class InfoLogger extends Logger { // 覆盖抽象字段 override var LogLevel: String = "INFO" }
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
。
trait
有默认的无参构造器,可是不支持有参构造器。一个类混入多个特质后初始化顺序应该以下:
// 示例 class Employee extends Person with InfoLogger with ErrorLogger {...}
更多大数据系列文章能够参见 GitHub 开源项目: 大数据入门指南