7、scala面向对象-继承
时间:2019-06-12
本文章向大家介绍7、scala面向对象-继承,主要包括7、scala面向对象-继承使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。
一、继承
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
原文地址:https://www.cnblogs.com/weiyiming007/p/11011096.html
- Python 3.6实现单博主微博文本、图片及热评爬取
- 用Django实现一个可运行的区块链应用
- Python的dict实现原理及与Java的比较探究
- 关于位域如何节省内存(C++)
- mysql的小知识点(关于数据库的导入导出 对于windows)
- Python网络编程中的套接字名和DNS解析
- hdu 4009 Transfer water(最小型树图)
- NumPy二元运算的broadcasting机制
- md5算法原理一窥(其一)
- 实现属于自己的TensorFlow(一) - 计算图与前向传播
- 基于Sanic的微服务基础架构
- hdu 3038 How Many Answers Are Wrong ( 带 权 并 查 集 )
- Java 基础知识点(必知必会其二)
- Java 基础知识点(必知必会其一)
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- Linux平台安装redis及redis扩展的方法
- Linux环境下快速搭建ftp服务器方法介绍
- Linux通过匿名管道进行进程间通信
- shell判断一个变量是否为空方法总结
- 基于ubuntu16 Python3 tensorflow(TensorFlow环境搭建)
- Linux中的iostat命令使用教程
- 深入解读Linux进程函数fork(),vfork(),execX()
- Linux centos下设置定时备份任务的方法步骤
- Linux 中如何切换相同程序的不同版本
- Linux中查看指定文件夹内各个子文件夹内的文件数量
- Linux系统中 /etc/fstab 文件的深入解读
- bash命令使用详解
- 实例详解Linux 中的命令链接操作符
- Python 为什么不支持 switch 语句?
- centos安装jdk1.8时出现没有/lib/ld-linux.so.2:这个文件的原因分析