Scala中的override

override是覆盖的意思,在很多语言中都有,在scala中,override是非常常见的,在类继承方面,它和java不一样,不是可写可不写的了,而是必须写的。如果不写而覆盖了对应的属性或者方法的话,编译器就会报错了。今天把scala中的override的各种地方都整理了一遍,以方便以后翻阅。

基础用法

/*
基本的override特性
*/
class A {
val nameVal = "A"
var nameVar = "A" def foo: String = {
"A.foo"
}
} class B extends A {
override val nameVal = "B"
//override var nameVar = "B" "variable nameVar cannot override a mutable variable"
override def foo: String = {
"B.foo"
}
} val b1 = new B
b1.foo
b1.nameVal
b1.nameVar val b2 : A = new B
b2.foo
b2.nameVal
b2.nameVar = "B"
b2.nameVar 输出: defined class A defined class B b1: B = B@9825fab
res0: String = B.foo
res1: String = B
res2: String = A b2: A = B@c46c4a1
res3: String = B.foo
res4: String = B
b2.nameVar: String = B
res5: String = B

当一个类extends另外一个类的时候,override的规则基本如下:

  • 子类中的方法要覆盖父类中的方法,必须写override(参见foo)
  • 子类中的属性val要覆盖父类中的属性,必须写override(参见nameVal)
  • 父类中的变量不可以覆盖(参见nameVar)

在抽象类中可以不用写override

/*
trait的extent不需要override
*/
trait T {
def foo : String
def bar : String
} class TB extends T {
def foo: String = {
"TB.foo"
} def bar: String = "TB.bar"
} val tb = new TB
tb.foo
tb.bar trait TT extends T {
def bar :String = "TT.bar"
} class TTB extends TT {
def foo: String = "TTB.foo"
}
val ttb = new TTB
ttb.foo
ttb.bar 输出: defined trait T defined class TB tb: TB = TB@2fb497ea
res6: String = TB.foo
res7: String = TB.bar defined trait TT defined class TTB ttb: TTB = TTB@346c06af
res8: String = TTB.foo
res9: String = TT.bar

T是特性类,它定义了两个抽象方法,foo和bar。TB的类继承和实现了T特性类,这个时候,TB类中的foo和bar前面的override是可写可不写的。这里初步看下TB类中的foo和bar前面的override写和不写感觉都一样,但是一旦有钻石结构的类继承,这个override的作用就体现出来了。这个我们后续说。

TT和TTB的例子也是说明了下trait继承trait是不需要使用override的。

abstrct class 也不需要使用override

/*
abstrct class 不需要override
*/
abstract class PA(name: String) {
def hello: String
} class PB(name: String) extends PA(name) {
def hello : String = s"hello ${name}"
} val pb = new PB("yejianfeng")
pb.hello 输出: defined class PA defined class PB pb: PB = PB@62840167
res10: String = hello yejianfeng abstract class和trait的特性主要是在是否有构造参数,在override方面都是一样的。

钻石结构

所谓的钻石结构就是一个菱形的结构,一个基类,两个子类,最后一个类又继承这两个子类。那么如果这两个子类都包含一个基类的方法,那么最后的这个类也有这个方法,选择继承那个子类呢?

/*
钻石结构
*/
trait Animal {
def talk: String
} trait Cat extends Animal {
def talk: String = "I am Cat"
} trait Monkey extends Animal {
def talk: String = "I am monkey"
} trait Dog extends Animal {
override def talk: String = "I am Dog"
} val kittyDog = new Cat with Dog
kittyDog.talk class MonkeyCat extends Monkey with Cat {
override def talk: String = "I am monkeyCat"
} val monkeyCat = new MonkeyCat
monkeyCat.talk 输出: defined trait Animal defined trait Cat defined trait Monkey defined trait Dog kittyDog: Cat with Dog = $anon$1@5378ef6d
res11: String = I am Dog defined class MonkeyCat monkeyCat: MonkeyCat = MonkeyCat@1e444ce6
res12: String = I am monkeyCat

在这个例子中,Animal是基类,Cat和Dog是子类,kittyDog是继承了Cat和Dog,那么kittyDog里面的talk使用的是Cat和Dog中有标示override的那个方法。这个时候override的作用就体现出来了。

参数复写使用override

我们可以直接在构造函数里面使用override重写父类中的一个属性。我理解这个更多是语法糖的一个功能。

/*
参数复写
*/
class Person(val age : Int){
val name = "no name"
} class XiaoMing(age: Int, override val name: String) extends Person(age){ }
val xiaoming = new XiaoMing(12, "xiaoming")
xiaoming.name 输出:
defined class Person defined class XiaoMing xiaoming: XiaoMing = XiaoMing@2eef0f3c
res13: String = xiaoming

总结

scala中的override基本是强制性的。这个我比较赞同,这样就减少了思维逻辑的负担,看到一个类中的一个方法的时候,就明白了这个方法是否是覆写父类的方法。但是感觉由于scala类的继承的灵活性,比如钻石结构里面,要知道最终的类使用的方法是什么,就需要了解每个父类的情况,这个还是有点纠结的。

Scala中的override的更多相关文章

  1. 第2节 Scala中面向对象编程:7、继承的概念以及override和super关键字;8、isInstanceOf 和 asInstanceOf关键字

    6.3.   Scala面向对象编程之继承 6.3.1.     Scala中继承(extends)的概念 Scala 中,让子类继承父类,与 Java 一样,也是使用 extends 关键字: 继承 ...

  2. scala中的面向对象定义类,构造函数,继承

    我们知道scala中一切皆为对象,函数也是对象,数字也是对象,它是一个比java还要面向对象的语言. 定义scala的简单类 class Point (val x:Int, val y:Int) 上面 ...

  3. scala入门教程:scala中的面向对象定义类,构造函数,继承

    我们知道scala中一切皆为对象,函数也是对象,数字也是对象,它是一个比java还要面向对象的语言. 定义scala的简单类 class Point (val x:Int, val y:Int) 上面 ...

  4. scala中的抽象类

    scala中也有和java,c#类似的抽象类,抽象类会有部分实现,也有没有实现的方法定义.抽象类最大的特征是不能直接实例化.下面我们看个例子. abstract class Animal { def ...

  5. Scala 中 构造函数,重载函数的执行顺序

    在调试scala在线开发教程(http://www.imobilebbs.com/wordpress/archives/4911)的过程中看到了以下代码,但是这段代码无论怎么调试都无法成功. abst ...

  6. Scala 深入浅出实战经典 第57讲:Scala中Dependency Injection实战详解

    王家林亲授<DT大数据梦工厂>大数据实战视频 Scala 深入浅出实战经典(1-87讲)完整视频.PPT.代码下载:百度云盘:http://pan.baidu.com/s/1c0noOt6 ...

  7. Scala 中的函数式编程基础(三)

    主要来自 Scala 语言发明人 Martin Odersky 教授的 Coursera 课程 <Functional Programming Principles in Scala>. ...

  8. scala中的Type使用

    trait Base { val name: String } case class S( name: String, age: Int ) extends Base case class F( na ...

  9. Programming In Scala笔记-第十一章、Scala中的类继承关系

    本章主要从整体层面了解Scala中的类层级关系. 一.Scala的类层级 在Java中Object类是所有类的最终父类,其他所有类都直接或间接的继承了Object类.在Scala中所有类的最终父类为A ...

随机推荐

  1. 实现ajax的步骤

    实现ajax的步骤 1.创建xmlHttprequest对象 function createXmlHttpRequest(){ var xmlHttp; try{ //Firefox, Opera 8 ...

  2. Leetcode题解(29)

    93. Restore IP Addresses 题目 分析:多重循环,判断小数点合适的位置 代码如下(copy网上) class Solution { public: vector<strin ...

  3. C# linq左连接与分组

    1.左连接使用DefaultIfEmpty(): 2.分组时候判断newper.FirstOrDefault() == null ? null: newper.ToList()这个经常出错误,如果不判 ...

  4. Flex 布局实例

    如图: 代码如下: <!DOCTYPE HTML> <html> <meta charset="utf-8"> <head> < ...

  5. Thinkphp5 模型 验证器执行顺序问题

    Thinkphp5把模型的验证规则归为一个验证器,这种做法,不知到符不符合大家的心意,反正楼主是比较不爽的 楼主更倾向于tp3.2的验证规则直接写在模型里面,毕竟你的验证规则一般而言是针对模型来验证的 ...

  6. 数据结构--汉诺塔递归Java实现

    /*汉诺塔递归 * 1.将编号0-N-1个圆盘,从A塔座移动到B上面 * 2.将编号N的1个圆盘,从A移动到C上面 * 3.最后将B上面的N-1个圆盘移动到C上面 * 注意:盘子的编号从上到下1-N ...

  7. 【开源】canvas图像裁剪、压缩、旋转

    前言 前段时间遇到了一个移动端对图像进行裁剪.压缩.旋转的需求. 考虑到已有各轮子的契合度都不高,于是自己重新造了一个轮子. 关于图像裁剪.压缩 在HTML5时代,canvas的功能已经非常强大了,可 ...

  8. C#导出EXCEL没有网格线的解决方法

    今天在做项目时,通过流导出数据到Excel却不显示网格线,真是郁闷.上网查了好久才得一良方(注意<XML>标签中的代码): DataTable thisTable = DBHelper.G ...

  9. CLR类型设计之属性

    在之前的随笔中,我们探讨了参数,字段,方法,我们在开始属性之前回顾一下,之前的探讨实际上串联起来就是OOP编程的思想,在接下来的文章中,我们还会讨论接口(就是行为),举个例子:我们如果要做一个学生档案 ...

  10. Centos6.9安装vsftpd并配置多用户的方法

    本文介绍了Centos6.9安装vsftpd并配置多用户的方法,分享给大家,具体如下: 一.安装vsftpd ? 1 2 3 4 5 6 7 8 #安装vsftpd yum -y install vs ...