Scala入门系列(七):面向对象之继承
extends
与Java一样,也是使用extends关键字,使用继承可以有效复用代码
class Person {
private var name = "leo"
def getName = name
}
class Student extends Person {
private var score = "A"
def getScore = score
}
defined class Person
defined class Student
scala> val s = new Student
s: Student = Student@53432aef
scala> s.getName
res27: String = leo
子类可以覆盖父类的field和method;但如果父类用final修饰,则该类是无法被继承的,如果field和method用final修饰,则field和method是无法被覆盖的。
override & super
跟Java一样,在Scala中,如果子类要覆盖一个父类中的非抽象方法,则必须要使用override关键字。
但再覆盖之后,如果我们需要在子类中调用父类被覆盖的方法该怎么办呢?那就可以使用super关键字,显式地指定要调用的父类方法。
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" + super.getName
}
在Scala中,子类可以覆盖父类的val field,而且子类的val field还可以覆盖父类val field的getter方法;只要在子类中使用override关键字。
class Person {
val name: String = "Person"
def age: Int = 0
}
class Student extends Person {
override val name: String = "leo"
override val age: Int = 30
}
defined class Person
defined class Student
scala> val s = new Student
// 测试覆盖了父类的getter方法
scala> s.age
res28: Int = 30
isInstanceOf & asInstanceOf
使用isInstanceOf 和 asInstanceOf可以安全的做到父类与子类对象之间类型的转化。
首先,需要使用isInstanceOf判断对象是否是指定类的对象,如果是的话,则可以使用asInstanceOf将对象转换为指定类型。
class Person
class Student extends Person
val p: Person = new Student
var s: Student = null
if (p.isInstanceOf[Student]) s = p.asInstanceOf[Student]
scala> s
res31: Student = Student@7ebfe01a
注:
- 如果对象时null,则isInstanceOf返回false,asInstanceOf返回null
- 如果没有用isInstanceOf先判断就直接asInstanceOf转换,则可能抛出异常
getCalss & classOf
isInstanceOf 只能判断出对象是否是指定类以及其子类的对象,而不能精确判断出,对象就是指定类的对象。
使用getCalss 和 classOf 就可以精确判断。
class Person
class Student extends Person
val p: Person = new Student
scala> p.getClass == classOf[Person]
res32: Boolean = false
scala> p.getClass == classOf[Student]
res33: Boolean = true
protected关键字
跟Java一样,Scala中同样可以使用protected关键字来修饰field和method,这样在子类中就不需要super关键字,直接就可以访问。
还可以使用protected[this],则只能在当前子类对象中访问父类的field和method,无法在其他子类对象中访问,跟private[this]一样。
class Person {
protected var name: String = "leo"
protected[this] var hobby: String = "game"
}
class Student extends Person {
def sayHelllo = println("hello, " + name)
def makeFriends(s: Student) {
println("my hobby is " + hobby + ",your hobby is " + s.hobby)
}
}
// 因为父类中使用了protected[this],所以这里s.hobby访问出错
<console>:19: error: value hobby is not a member of Student
println("my hobby is " + hobby + ",your hobby is " + s.hobby)
^
调用父类的constructor
在Scala中,每个类可以有一个主constructor和任意多个辅constructor,而每个辅助constructor的第一行都必须是调用其他辅助constructor或者是主constructor,因此子类的辅助constructor是一定不可能直接调用父类的constructor的。
只能在子类的主constructor中调用父类的constructor,采用以下语法:
class Person(val name: String, val age: Int)
/* 注意,如果是父类接收的参数,比如name和age,子类中接收时,就不要用任何val或者var来修饰,否则会被认为是子类覆盖父类的field。*/
class Student(name: String, age: Int, var score: Double) extends Person(name, age) {
def this(name: String) {
this(name, 0, 0)
}
def this(age: Int) {
this("sparks", age, 0)
}
}
defined class Person
defined class Student
scala> val s = new Student("sparks")
s: Student = Student@4c3fcbe7
scala> val s = new Student(20)
s: Student = Student@7852ab30
匿名内部类
在Scala中,匿名内部类是非常强大且常见的,Spark源码中也大量使用了这种匿名内部类。
定义一个类的没有名字的子类,并直接创建其对象,然后将该对象的引用赋予一个变量,之后甚至可以将该对象传递给其他函数。
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)
}
// 测试
scala> greeting(p)
Hi, I'm leo
抽象类
跟Java相似,一个类中如果有一个抽象方法,那么类就必须用abstract来声明类,此时抽象类不可以实例化。
在子类覆盖抽象类的抽象方法时,不需要使用override 关键字
abstract class Person(val name: String){
def sayHello: Unit
}
class Student(name: String) extends Person(name) {
def sayHello: Unit = println("Hello, " + name)
}
如果在抽象类中定义了field,但没有给出初始值,则此field为抽象field
抽象field意味着:scala会根据自己的规则,为var或val类型的field生成对应的getter和setter方法,但是父类中是没有该field的,子类必须覆盖field,不需要使用override关键字。
abstract class Person{
val name: String
}
// 子类必须要覆盖抽象field
class Student extends Person{
val name: String = "leo"
}
Scala入门系列(七):面向对象之继承的更多相关文章
- Scala入门系列(八):面向对象之trait
基础知识 1 将trait作为接口使用 此时Trait就与Java中的接口非常类似,不过注意,在Scala中无论继承还是trait,统一都是extends关键字. Scala跟Java 8前一样不支持 ...
- Scala入门系列(九):函数式编程
引言 Scala是一门既面向对象,又面向过程的语言,Scala的函数式编程,就是Scala面向过程最好的佐证.也真是因此让Scala具备了Java所不具备的更强大的功能和特性. 而之所以Scala一直 ...
- Scala入门系列(六):面向对象之object
object object相当于class的单个实例,类似于Java中的static,通常在里面放一些静态的field和method. 第一次调用object中的方法时,会执行object的con ...
- Scala入门系列(五):面向对象之类
定义类 // 定义类,包含field以及method class HelloWorld { private var name = "Leo" def sayHello() { pr ...
- Scala入门系列(一):基础语法
Scala基础语法 Scala与JAVA的关系 Scala是基于Java虚拟机,也就是JVM的一门编程语言,所有Scala的代码都需要经过编译为字节码,然后交由Java虚拟机来运行. 所以Scala和 ...
- Scala入门系列(四):Map & Tuple
Map 创建Map // 创建一个不可变的Map scala> val ages = Map("Leo" -> 30, "Sparks" -> ...
- Scala入门系列(十):函数式编程之集合操作
1. Scala的集合体系结构 Scala中的集合体系主要包括(结构跟Java相似): Iterable(所有集合trait的根trait) Seq(Range.ArrayBuffer.List等) ...
- Scala入门系列(十一):模式匹配
引言 模式匹配是Scala中非常有特色,非常强大的一种功能. 类似于Java中的switch case语法,但是模式匹配的功能要比它强大得多,switch只能对值进行匹配,但是Scala的模式匹配除了 ...
- Scala入门系列(十二):隐式转换
引言 Scala提供的隐式转换和隐式参数功能,是非常有特色的功能.是Java等编程语言所没有的功能.它可以允许你手动指定,将某种类型的对象转换成其他类型的对象.通过这些功能可以实现非常强大而且特殊的功 ...
随机推荐
- position,display,float,overflow,margin,padding之间的相互影响
1.元素分为块级元素和行内元素, 块级元素可以设置宽高,会自动换行,并且会发生相邻margin的合并问题.行内元素设置宽和高无效,以水平方向排列,(行内元素,绝对定位,浮动元素不会发生外边距合并)并且 ...
- 动态加载js,css(项目中需要的)
最近做的一个项目需要加入百度统计,大家都知道百度统计在页面引用就是一坨js,实现方法很简单引用到页面就ok了. 那么问题来了,虽然我不知道百度统计的原理是啥,我的测试服引用了百度统计,百度统计账号里面 ...
- linux学习(七)环境变量、cp、mv、cat,less,more,head,tail
一.环境变量 环境变量其实就是$PATH: [root@iZ25lzba47vZ ~]# echo $PATH /usr/local/nginx/sbin:/usr/local/php/bin:/us ...
- 简单的计算100000以内的质数(JAVA实现)
public class TestPrimeNumber(int i){ public static void main(String[] args) { long start = System.cu ...
- Problem N
Problem Description The cows have purchased a yogurt factory that makes world-famous Yucky Yogurt. O ...
- Hadoop部署配置文件
为了方便大家修,我把要修改的地方标红了,找到对应文件,复制粘贴过去就可以了 注:这个是我安装的Hadoop的配置,要根据我之前发的文章配置才行, 里面有一些东西比如文件夹名称,路径不一样,稍加修改也可 ...
- POI不同版本替换Word模板时的问题
一.问题描述 通过POI,把Word中的占位符替换为实际的值,以生成复杂结构的业务报告. 在POI 3.9上,功能正常.由于某些原因升级到POI 3.10.1后,项目组反馈说Word模板出错,无法生成 ...
- 阿里云centos配置postfix
1. 为了防止垃圾邮件,先到域名控制面板设置好. MX A 记录及TXT记录 其中TXT记录如下 @ spf1 a mx ~all 意思就是使用spf1协议,允许a记录和MX记录对应的IP,不允许 ...
- windows系统扩展C盘的工具推荐(解决了C盘和压缩卷不相邻无法扩展C盘问题)
1.下载分区工具 “分区助手3.0中文版” 下载地址:http://www.33lc.com/soft/14880.html 2.下载下来是一个压缩包,解压后运行安装程序. 3.安装完成后按以下步骤执 ...
- WPF 简易的喷泉效果
这两天领导让我做个喷泉的效果,要把一个个UserControl从一个位置喷出,然后,最后落在最终需要在的位置. 喷泉效果说白了,就是两个步骤:1.放大,从0放大到需要的倍数:2.缩小,平移,从放大的倍 ...