一、继承

1、extends

Scala中,让子类继承父类,与Java一样,也是使用extends关键字
继承就代表,子类可以从父类继承父类的field和method;然后子类可以在自己内部放入父类所没有,
子类特有的field和method;使用继承可以有效复用代码
子类可以覆盖父类的field和method;但是如果父类用final修饰,field和method用final修饰,则该类是无法被继承的,
field和method是无法被覆盖的; ###
scala> :paste
// Entering paste mode (ctrl-D to finish) class Person {
private var name = "leo"
def getName = name
} class Student extends Person {
private var score = "A"
def getScore = score
} // Exiting paste mode, now interpreting. defined class Person
defined class Student
warning: previously defined object Person is not a companion to class Person.
Companions must be defined together; you may wish to use :paste mode for this. scala> val s = new Student
s: Student = Student@985696 scala> s.getName
res11: String = leo scala> s.getScore
res12: String = A scala> s.getName
res13: String = leo scala> s.name
<console>:14: error: value name is not a member of Student
s.name
^

2、override和super

//Scala中,如果子类要覆盖一个父类中的非抽象方法,则必须使用override关键字;

//override关键字可以帮助我们尽早地发现代码里的错误,比如:override修饰的父类方法的方法名我们拼写错了;比如要覆盖的父类方法的参数我们写错了;等等

//此外,在子类覆盖父类方法之后,如果我们在子类中就是要调用父类的被覆盖的方法呢?
那就可以使用super关键字,显式地指定要调用父类的方法 ###
scala> :paste
// Entering paste mode (ctrl-D to finish) class Person {
private var name = "leo"
def getName = name
} class Student extends Person {
private var score = "A"
def getScore = score
override def getName = "Hi, I'm a student, my name is " + super.getName
} // Exiting paste mode, now interpreting. defined class Person
defined class Student
warning: previously defined object Person is not a companion to class Person.
Companions must be defined together; you may wish to use :paste mode for this. scala> val s = new Student
s: Student = Student@618ad2aa scala> s.getName
res15: String = Hi, I'm a student, my name is leo

3、override field

// Scala中,子类可以覆盖父类的val field,而且子类的val field还可以覆盖父类的val field的getter方法;只要在子类中使用override关键字即可

###
scala> :paste
// Entering paste mode (ctrl-D to finish) class Person {
val name: String = "Person"
def age: Int = 0
} class Student extends Person {
override val name: String = "leo"
override val age: Int = 30
} // Exiting paste mode, now interpreting. defined class Person
defined class Student
warning: previously defined object Person is not a companion to class Person.
Companions must be defined together; you may wish to use :paste mode for this. scala> val p = new Person
p: Person = Person@71d9cb05 scala> p.name
res16: String = Person scala> p.age
res17: Int = 0 scala> val s = new Student
s: Student = Student@1ac4ccad scala> s.name
res18: String = leo scala> s.age
res19: Int = 30

4、isInstanceOf和asInstanceOf

// 如果我们创建了子类的对象,但是又将其赋予了父类类型的变量。则在后续的程序中,我们又需要将父类类型的变量转换为子类类型的变量,应该如何做?
// 首先,需要使用isInstanceOf判断对象是否是指定类的对象,如果是的话,则可以使用asInstanceOf将对象转换为指定类型
// 注意,如果对象是null,则isInstanceOf一定返回false,asInstanceOf一定返回null
// 注意,如果没有用isInstanceOf先判断对象是否为指定类的实例,就直接用asInstanceOf转换,则可能会抛出异常 ###
scala> class Person
defined class Person
warning: previously defined object Person is not a companion to class Person.
Companions must be defined together; you may wish to use :paste mode for this. scala> class Student extends Person
defined class Student scala> val p: Person = new Student
p: Person = Student@f324455 scala> var s: Student = null
s: Student = null scala> if (p.isInstanceOf[Student]) s = p.asInstanceOf[Student]

5、getClass和classOf

// isInstanceOf只能判断出对象是否是指定类以及其子类的对象,而不能精确判断出,对象就是指定类的对象
// 如果要求精确地判断对象就是指定类的对象,那么就只能使用getClass和classOf了
// 对象.getClass可以精确获取对象的类,classOf[类]可以精确获取类,然后使用==操作符即可判断 ###
scala> class Person
defined class Person scala> class Student extends Person
defined class Student scala> val p: Person = new Student
p: Person = Student@77468bd9 scala> p.isInstanceOf[Person]
res0: Boolean = true scala> p.getClass == classOf[Person]
res1: Boolean = false scala> p.getClass == classOf[Student]
res2: Boolean = true

6、使用模式匹配进行类型判断

// 但是在实际开发中,比如spark的源码中,大量的地方都是使用了模式匹配的方式来进行类型的判断,这种方式更加地简洁明了,
而且代码得可维护性和可扩展性也非常的高
// 使用模式匹配,功能性上来说,与isInstanceOf一样,也是判断主要是该类以及该类的子类的对象即可,不是精准判断的 ###
scala> class Person
defined class Person scala> class Student extends Person
defined class Student scala> val p: Person = new Student
p: Person = Student@192d3247 scala> p match {
| case per: Person => println("it's Person's object")
| case _ => println("unknown type")
| }
it's Person's object

7、protected

// 跟java一样,scala中同样可以使用protected关键字来修饰field和method,这样在子类中就不需要super关键字,直接就可以访问field和method
// 还可以使用protected[this],则只能在当前子类对象中访问父类的field和method,无法通过其他子类对象访问父类的field和method ###
scala> :paste
// Entering paste mode (ctrl-D to finish) class Person {
protected var name: String = "leo"
} class Student extends Person {
def makeFriends(s: Student) {
println("Hi, my name is " + name + ", your name is " + s.name)
}
} // Exiting paste mode, now interpreting. defined class Person
defined class Student scala> val s1 = new Student
s1: Student = Student@63355449 scala> val s2 = new Student
s2: Student = Student@1ab3a8c8 scala> s1.makeFriends(s2)
Hi, my name is leo, your name is leo scala> :paste
// Entering paste mode (ctrl-D to finish) class Person {
protected[this] val name: String = "leo"
} class Student extends Person {
def makeFriends(s: Student) {
println("my name is " + name + ", your name is " + s.name)
}
} // Exiting paste mode, now interpreting. <console>:22: error: value name is not a member of Student
println("my name is " + name + ", your name is " + s.name)
^

8、调用父类的constructor

// Scala中,每个类可以有一个主constructor和任意多个辅助constructor,而每个辅助constructor的第一行都必须是调用其他辅助constructor或者是主constructor;
因此子类的辅助constructor是一定不可能直接调用父类的constructor的
// 只能在子类的主constructor中调用父类的constructor,以下这种语法,就是通过子类的主构造函数来调用父类的构造函数
// 注意!如果是父类中接收的参数,比如name和age,子类中接收时,就不要用任何val或var来修饰了,否则会认为是子类要覆盖父类的field ###
scala> :paste
// Entering paste mode (ctrl-D to finish) class Person(val name: String, val age: Int) class Student(name: String, age: Int, var score: Double) extends Person(name, age) {
def this(name: String) {
this(name, 0, 0.0)
}
def this(age: Int) {
this("leo", age, 0)
}
} // Exiting paste mode, now interpreting. defined class Person
defined class Student scala> val s = new Student("leo", 30, 100)
s: Student = Student@783a467b scala> s.name
res6: String = leo scala> s.age
res7: Int = 30 scala> s.score
res8: Double = 100.0 scala> val s2 = new Student("leo")
s2: Student = Student@6f204a1a scala> s2.name
res9: String = leo scala> s2.age
res10: Int = 0 scala> val s3 = new Student(30)
s3: Student = Student@3224a577 scala> s3.name
res11: String = leo scala> s3.age
res12: Int = 30 scala> s3.score
res13: Double = 0.0

9、匿名子类

// 在Scala中,匿名子类是非常常见,而且非常强大的。Spark的源码中也大量使用了这种匿名子类。
// 匿名子类,也就是说,可以定义一个类的没有名称的子类,并直接创建其对象,然后将对象的引用赋予一个变量。之后甚至可以将该匿名子类的对象传递给其他函数。 ###
scala> :paste
// Entering paste mode (ctrl-D to finish) class Person(protected val name: String) {
def sayHello = "Hello, I'm " + name
} val p = new Person("leo") {
override def sayHello = "Hi, I'm " + name
} def greeting(p: Person {def sayHello: String}) {
println(p.sayHello)
} // Exiting paste mode, now interpreting. defined class Person
p: Person = $anon$1@3f270e0a
greeting: (p: Person{def sayHello: String})Unit scala> greeting(p)
Hi, I'm leo

10、抽象类

// 如果在父类中,有某些方法无法立即实现,而需要依赖不同的子来来覆盖,重写实现自己不同的方法实现。此时可以将父类中的这些方法不给出具体的实现,
只有方法签名,这种方法就是抽象方法。
// 而一个类中如果有一个抽象方法,那么类就必须用abstract来声明为抽象类,此时抽象类是不可以实例化的
// 在子类中覆盖抽象类的抽象方法时,不需要使用override关键字 ###
scala> :paste
// Entering paste mode (ctrl-D to finish) abstract class Person(val name: String) {
def sayHello: Unit
} class Student(name: String) extends Person(name) {
def sayHello: Unit = println("Hello, " + name)
} // Exiting paste mode, now interpreting. defined class Person
defined class Student scala> val p = new Person
<console>:11: error: class Person is abstract; cannot be instantiated
val p = new Person
^ scala> val s = new Student("leo")
s: Student = Student@368239c8 scala> s.sayHello
Hello, leo

11、抽象field

// 如果在父类中,定义了field,但是没有给出初始值,则此field为抽象field
// 抽象field意味着,scala会根据自己的规则,为var或val类型的field生成对应的getter和setter方法,但是父类中是没有该field的
// 子类必须覆盖field,以定义自己的具体field,并且覆盖抽象field,不需要使用override关键字 ###
scala> :paste
// Entering paste mode (ctrl-D to finish) abstract class Person {
val name: String
} class Student extends Person {
val name: String = "leo"
} // Exiting paste mode, now interpreting. defined class Person
defined class Student scala> val s = new Student
s: Student = Student@1794d431 scala> s.name
res1: String = leo

7、scala面向对象-继承的更多相关文章

  1. scala 面向对象之 继承

    scala 面向对象之 继承 scala   1.extends Scala中,让子类继承父类,与Java一样,也是使用extends关键字 继承就代表,子类可以从父类继承父类的field和metho ...

  2. Scala 面向对象(六):面向对象的特征二:继承 (一)

    1 Scala继承的基本语法 class 子类名 extends 父类名 { 类体 } class Person { var name : String = _ var age : Int = _ d ...

  3. Spark 3000门徒第二课scala面向对象总结

    昨晚听了王家林老师3000门徒spark系列课程的第二课,讲述了scala面向对象知识,并且带着过了一遍Spark核心类:SparkContent,RDD的代码,下面写一下心得: RDD是抽象类,实现 ...

  4. 7. Scala面向对象编程(中级部分)

    7.1 包 7.1.1 看一个应用场景 现在有两个程序员共同开发一个项目,程序员xiaoming希望定义一个类取名Dog,程序员xiaohong也想定一个类也叫Dog,两个程序员还为此吵了起来,该怎么 ...

  5. Scala 面向对象(八):特质(接口) 一

    1 Scala接口的介绍 从面向对象来看,接口并不属于面向对象的范畴,Scala是纯面向对象的语言,在Scala中,没有接口. Scala语言中,采用特质trait(特征)来代替接口的概念,也就是说, ...

  6. OC面向对象—继承

    OC面向对象—继承 一.基本概念 程序的世界和人类的“对象”世界在思想上是没有设么区别的,富二代继承了父母,自然就拥有了父母拥有的所有资源,子类继承了父类同样就拥有了父类所有的方法和属性(成员变量). ...

  7. scala的继承

    package com.test.scala.test /** * 模拟java的继承,扩展类 */ abstract class ExtendClass(val des:String) { def ...

  8. 面向对象继承实例(a如何继承b)

    <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title> ...

  9. js面向对象继承

    前言 最近看到js面向对象这章节了,主要学习了原型和面向对象继承关系,为了梳理自己的知识逻辑,特此记录. js的面向对象 先说说我目前了解的js创建对象方法 1.写一个函数,然后通过new创建对象 2 ...

随机推荐

  1. zookeeper 实战案例分享:cruator客户端编程

    上两篇介绍了zookeeper服务器端的安装和配置,今天分享下利用zookeeper客户端编程来实现配置文件的统一管理,包括文件添加.删除.更新的同步. 比如,连接数据库信息的配置文件,一般每个应用服 ...

  2. P4965 薇尔莉特的打字机

    题目 P4965 薇尔莉特的打字机 快到十二点了正在颓废突然发现了一道好题 虽然毒瘤,但确实是容斥原理的好题啊,做法也特别巧妙(标程 思路 题目大意(怕自己突然忘) n个初始字符,m个操作(加入或删除 ...

  3. java 遍历数组的几种方式

    本文总结自: https://www.cnblogs.com/hellochennan/p/5373186.html 1. 传统方式 非常简单的for循环 int[] a = {1, 2, 3, 4} ...

  4. SrpingCloud 之SrpingCloud config分布式配置中心实时刷新

    默认情况下是不能及时获取变更的配置文件信息 Spring Cloud分布式配置中心可以采用手动或者自动刷新 1.手动需要人工调用接口   监控中心 2.消息总线实时通知  springbus 动态刷新 ...

  5. GridView行中按钮的使用

    转载自:http://blog.csdn.net/hongdi/article/details/6455947 GridView行中按钮的使用 在web项目的过程中,特别是开发ASP.NET应用程序, ...

  6. tomcat使用JDNI配置信息和使用信息。用于JDBC连接池

    JNDI: JNDI(java Naming and Directory Interface),java命名和目录接口.JNDI的作用就是:在服务器上配置资源,然后通过统一的方式来获取配置的资源 在t ...

  7. linux命令学习笔记(57):ss命令

    ss是Socket Statistics的缩写.顾名思义,ss命令可以用来获取socket统计信息,它可以显示和netstat 类似的内容.但ss的优势在于它能够显示更多更详细的有关TCP和连接状态的 ...

  8. 【二叉树的递归】03判断二叉树中有没有和为给定值的路径【Path Sum】

    ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ 给定一个二叉树和一个和,判断这个树 ...

  9. C++函数重载详解

    我们在开瓶瓶罐罐的时候,经常会遭遇因各种瓶口规格不同而找不到合适的工具的尴尬.所以有时候就为了开个瓶,家里要备多种规格的开瓶器.同样是开个瓶子嘛,何必这么麻烦?于是有人发明了多功能开瓶器,不管啤酒瓶汽 ...

  10. [转]django 日志logging的配置以及处理

    http://davidbj.blog.51cto.com/4159484/1433741 日志在程序开发中是少不了的,通过日志我们可以分析到错误在什么地方,有什么异常.在生产环境下有很大的用途.在J ...