一.本章要点

  • extends,final关键字和Java一样
  • 重写方法时必须使用override
  • 只有主构造器可以调用超类的构造器
  • 可以重写字段

二.扩展类

  Scala扩展类和Java一样(使用extends关键字),也可以将类声明为final让它不能被扩展,也可以将单个方法或字段声明为final,确保不能重写。

  注:Java中final是不可变的(相当于Scala中的val),但是Scala中是不能被扩展

三.重写方法

  使用override修饰符:

  1. public class Person{
  2. override def toString=....
  3. }

  override的作用在多个常见情况下给出有用的错误提示,如:

      • 拼错重写的方法名
      • 在新方法中使用了错误的参数类型
      • 在超类中引用新的方法,但是这个方法与子类方法相抵触(易违约基类问题,超类的修改无法在不检查所有子类的前提下被验证)

  在Scala中调用超类的方法和Java完全一样,使用super关键字

四.类型检查和转换

  isInstanceOf:测试某个对象是否属于某个给定的类,可以用isInstanceOf方法,如果为true就可以用asInstanceOf方法将引用转换为子类的引用。

  例:

  1. //如果p指向的是Employee类及其子类(比如Manager)的对象,则p.isInstanceOf[Employee]将会成功。
  2. //如果p是null,则p.isInstanceOf[Employee]返回false,p.asInstanceOf[Employee]将返回null
  3. //如果p不是一个Employee,则p.asInstanceOf[Employee]将抛出异常
  4. if(p.isInstanceOf[Employee]){
  5. val s=p.asInstanceOf[Employee]
  6.  
  7. }

  classOf(在scala.Predef中,可以自动导入):测试某个对象是某个类的对象但又不是其子类。

  例:

  1. if(p.getClass==classOf[Employee])

五.受保护字段和方法

  将字段或方法声明为protected,这样的成员可以被任何子类访问,但不能从其他位置看到。

  注:与Java不同,protected的成员对于类所属的包而言,是不可见的;  

    Scala还停工了protected[this]的变体,把访问权限限定在当前的对象

六.超类的构造

  辅助构造器永远不可能直接调用超类的构造器,子类的辅助构造器最终都会调用主构造器,只有主构造器可以调用超类的构造器。

  主类构造器和类定义交织在一起,调用超类构造器也是交织在一起。

  例:

  1. //Employee类的三个参数有两个传递给了超类
  2. class employee(name:String,age:Int,val salary:Double) extends Person(name,age)

  注:在Scala构造器中,不能调用super(params)[Java可以]

    Scala可以扩展Java类,这种情况必须调用Java类的某一个构造方法

七.重写字段

  子类可以重写父类的字段。

  注:

      • def只能重写另一个def
      • val只能重写另一个val或不带参数的def
      • val只能重写另一个抽象的var

  

八.匿名子类

  和Java一样,可以通过包含带有定义或重写代码块的方式创建一个匿名的子类,例:

  1. //这会创建一个结构类型的对象,可以作为参数类型的定义
  2. val alien=new Person("Fred"){
  3.  
  4. def greeting="xxxxxxx"
  5.  
  6. }
  7.  
  8. def meet(p:Person(def greeting:String)){}

九.抽象类

  和Java一样,可以用abstract关键字标记不能被实例化的类(通常因为某个或某几个方法没有被完整定义)。

  例:

 

  1. abstract class Employee(val name:String){
  2. def id:Int //没有方法体——抽象方法
  3. }

  注:

    Scala中抽象方法不用使用abstract关键字,只需省略方法体就行;  

    只要类中有一个抽象方法,这个类就需要是抽象类,类必须声明abstract;  

    子类中重写超类的抽象方法时不需要使用override关键字

十.抽象字段

  即没有初始值的字段。

  例:

  1. abstract class Person{
  2. val id:Int //没有初始化——带有抽象的getter方法的抽象字段
  3. var name:String //带有抽象的getter和setter方法
  4.  
  5. }

  注:子类必须提供具体的字段;

    子类重写超类的字段不需要使用override

  1. //具体的id和name属性
  2. class Employee(var id:Int) extends Person{
  3. var name=""
  4.  
  5. }
  6. //通过匿名类型定制抽象字段
  7. val fred=new Person{
  8.  
  9. val id=1
  10. val name="Fred"
  11. }

十一.构造顺序和提前定义

  在子类中重写val并且在超类中的构造器使用该值有可能出现问题(因为超类的构造器先于子类的构造器运行,根本原因来自Java语言的设定——允许在超类的构造方法中调用子类的方法),如:

  

  1. class Creature{
  2. val range:Int=10
  3. val env:Array[Int] = new Array[Int](range)
  4.  
  5. }
  6.  
  7. class Ant extends Creature{
  8. override val range=2
  9.  
  10. }
  11. ......
  12. //长度为0
  13. val ant=new Ant
  14. println(ant.env.length)
  15. /××
  16. 过程如下:
  17. 1.Ant的构造器在构造自己之前先调用Creature的构造器;
  18. 2.Creature的构造器把它的range设为10;
  19. 3.Creature的构造器为了初始化env数组,调用range()取值器;
  20. 4.该方法被重写以输出(还未初始化)Ant类的range字段值;
  21. 5.range方法返回0(这是对象被分配空间时所有整型字段的初始值);
  22. 6.env被设为长度为0的数组;
  23. 7.Ant构造器继续执行,将其range设为2
  24. ××/

  注:因此构造器内不应该依赖val的值;

    解决方式:

      1.将val声明为final,这样很安全但并不灵活;

      2.在超类中将val声明为lazy,这样很安全但不高效;

      3.在子类中使用提前定义方法(可以在超类的构造器执行之前初始化子类的val字段),使用with关键字(提前定义的等号右侧只能引用之前已有的提前定义,而不能使用类中的其他字段或方法)

  1. class Ant extends{
  2. override val range=2
  3.  
  4. }with Creature

    可以使用-Xcheckinit编译器标志调式构造顺序    

十二.Scala继承等级

Scala继承等级

  注:

    AnyRef是Java或.Net虚拟机Object类的同义词;

    AnyVal和AnyRef都扩展于Any类,而Any类是整个继承等级的根节点;

    Any类中定义了isInstanceOf,asInstanceOf方法,相等性判断,哈希码的方法;

    AnyVal没有追加任何方法,他只是一个标记;

    AnyRef追加了来自Object类的监视方法wait和notify/notifyAll,同时提供了synchronized方法;

    所有Scala类都实现ScalaObject这个标记接口,该接口没有任何方法;

    Null值的唯一实例是null;

    Nothing类型没有实例  

十三.对象相等性

  Scala中AnyRef的eq方法检查两个引用是否指向同一个对象,AnyRef的equals方法调用eq。

  实现类的时候有需要时重写equals根据自己的逻辑判断,例:

  注:将方法定义为final,因为通常而言在子类中正确扩展相等性判断非常困难;

    确保定义的equals方法参数类型为Any ;

    再定义equals时,记得同时定义hashCode 

  1. final override def equals(other:Any){
  2.  
  3. val that=other.asInstanceOf(Item)
  4. if(that==null) false
  5. else description==that.description&&price==that.price
  6. }

  

十四.练习

Scala学习八——继承的更多相关文章

  1. Scala学习(八)---Scala继承

    Scala继承 摘要: 在本篇中,你将了解到Scala的继承与Java和C++最显著的不同.要点包括: 1. extends.final关键字和Java中相同 2. 重写方法时必须用override ...

  2. Scala学习(八)练习

    Scala中继承&练习 1. 扩展如下的BankAccount类,新类CheckingAccount对每次存款和取款都收取1美元的手续费 class BankAccount ( initial ...

  3. JavaScript之面向对象学习八(继承)

    简介:继承是OO语言中的一个最为人津津乐道的概念.许多OO语言都支持两种继承方式:接口继承和实现继承.接口继承只继承方法签名,而实现继承则继承实际的方法. 但是JS的函数并没有签名,所以在ECMASc ...

  4. scala学习笔记2(类,继承,抽象类)

    class Person{ // _ 是占位符; var name : String = _ val age : Int = 27 // private[this] 定义的内容无法外部使用,起到保护作 ...

  5. (C/C++学习笔记) 十八. 继承和多态

    十八. 继承和多态 ● 继承的概念 继承(inheritance): 以旧类为基础创建新类, 新类包含了旧类的数据成员和成员函数(除了构造函数和析构函数), 并且可以派生类中定义新成员. 形式: cl ...

  6. Scala 学习之路(九)—— 继承和特质

    一.继承 1.1 Scala中的继承结构 Scala中继承关系如下图: Any是整个继承关系的根节点: AnyRef包含Scala Classes和Java Classes,等价于Java中的java ...

  7. 【Todo】【读书笔记】大数据Spark企业级实战版 & Scala学习

    下了这本<大数据Spark企业级实战版>, 另外还有一本<Spark大数据处理:技术.应用与性能优化(全)> 先看前一篇. 根据书里的前言里面,对于阅读顺序的建议.先看最后的S ...

  8. Python Tutorial 学习(八)--Errors and Exceptions

    Python Tutorial 学习(八)--Errors and Exceptions恢复 Errors and Exceptions 错误与异常 此前,我们还没有开始着眼于错误信息.不过如果你是一 ...

  9. 【大数据】Scala学习笔记

    第 1 章 scala的概述1 1.1 学习sdala的原因 1 1.2 Scala语言诞生小故事 1 1.3 Scala 和 Java  以及 jvm 的关系分析图 2 1.4 Scala语言的特点 ...

随机推荐

  1. docker安装redis,并用配置启动

    1.拉取redis镜像 docker pull redis 2.创建redis本地配置文件 ①.去redis官网下载redis,获取redis.conf文件 ②.修改redis.conf文件相关配置, ...

  2. LVS之NAT模式

    目录: 网络环境 LVS服务器配置 LVS服务器添加ipvs规则 RS服务器配置 访问验证 抓包分析 注意事项 [网络环境] 服务器类型 IP 说明 lvs_vip 192.168.2.130 vip ...

  3. SRS之SrsHls::on_audio详解

    1. SrsHls::on_audio 将音频数据封装到 ts 文件中. /* * mux the audio packet to ts. * @param shared_audio, directl ...

  4. 我现在A函数开启事务,然后调用B函数,B函数中也开启了事务

    有一点要知道,就是MYSQL不支持事务嵌套.所以PHP再包装,也是一个事务 laravel的事务嵌套,就是一个栈.事务A开启事务(真实开启)事务B开启事务(只是标记,并非真实开启了事务)事务B提交事务 ...

  5. Android7.0 API变更

    Android N 除了提供诸多新特性和功能外,还对系统和 API 行为做出了各种变更. 本文重点介绍您应该了解并在开发应用时加以考虑的一些重要变更. 如果您之前发布过 Android 应用,请注意您 ...

  6. [go]结构体/接口

    接口惯用操作: 结构体构造方法返回接口类型 //定义服务器接口 type IServer interface{ Start() Stop() Serve() } type Server struct ...

  7. nginx配置, 启动命令, 反向代理配置

    2014年1月3日 13:52:07 喜欢这样的风格,干货 http://huoding.com/2013/10/23/290 -----------------下边是我自己的经验(windows)- ...

  8. unmix和conditional average:消混叠和条件均值

    unmix 该程序用来消除“像素混叠”.所谓像素混叠,是值在自然场景的图像中,边缘线成像到cmos的像素上时,某些像素会刚好跨在该边缘线上. 这样的像素特点就是,其R.G.B三色像素梯度值不一致.比如 ...

  9. Kaggle初体验之泰坦尼特生存预测

    Kaggle初体验之泰坦尼特生存预测 学习完了决策树的ID3.C4.5.CART算法,找一个试手的地方,Kaggle的练习赛泰坦尼特很不错,记录下 流程     首先注册一个账号,然后在顶部菜单栏Co ...

  10. Spring Boot Lombok配置

    Spring Boot Lombok配置 依赖添加 dependencies { annotationProcessor 'org.projectlombok:lombok:1.18.2' compi ...