scala 面向对象之 继承
scala 面向对象之 继承
scala
1.extends
Scala中,让子类继承父类,与Java一样,也是使用extends关键字
继承就代表,子类可以从父类继承父类的field和method;然后子类可以在自己内部放入父类所没有,子类特有的field和method;使用继承可以有效复用代码
class Person {private var name = "leo"def getName = name}class Student extends Person {private var score = "A"def getScore = score}
子类可以覆盖父类的field和method;但是如果父类用final修饰,则该类是无法被继承的,field和method用final修饰,field和method是无法被覆盖的
final class Person { //final embellish class Person it not be extendsprivate var name = "leo"def getName = name}class Student extends Person { //errorprivate var score = "A"def getScore = score}
eg:
scala> final class Person {
| private var name = "leo"
| def getName = name
| }
defined class Person
scala> class Student extends Person {
| private var score = "A"
| def getScore = score
| }
<console>:8: error: illegal inheritance from final class Person
class Student extends Person {
override 父类final修饰的字段 方法
class Person {private var name = "leo"final val age = 10; //final embellish field age it not be override by child classdef getName = name}class Student extends Person {private var score = "A"def getScore = scoreoverride val age = 20 //erroroverride def getName = "my name is" + super.getName}
eg:
scala> class Person {
| private var name = "leo"
| final val age = 10; //final embellish field age it not be override by child class
| def getName = name
| }
defined class Person
scala> class Student extends Person {
| private var score = "A"
| def getScore = score
| override val age = 20 //error
| override def getName = "my name is" + super.getName
| }
<console>:11: error: overriding value age in class Person of type Int(10);
value age cannot override final member
override val age = 20 //error
2.isInstanceOf 和 asInstanceOf
如果我们创建了子类的对象,但是又将其赋予了父类类型的变量。则在后续的程序中,我们又需要将父类类型的变量转换为子类类型的变量,应该如何做?
首先,需要使用isInstanceOf判断对象是否是指定类的对象,如果是的话,则可以使用asInstanceOf将对象转换为指定类型
注意,如果对象是null,则isInstanceOf一定返回false,asInstanceOf一定返回null
注意,如果没有用isInstanceOf先判断对象是否为指定类的实例,就直接用asInstanceOf转换,则可能会抛出异常
class Personclass Student extends Personval p: Person = new Studentvar s: Student = null //note:isInstanceOf and asInstanceOf use[] instead of ()if (p.isInstanceOf[Student]) s = p.asInstanceOf[Student]
但,isInstanceOf只能判断是否为某个类或者某个类的子类,无法精确判断。
eg:
scala> print(p.isInstanceOf[Student])
true
scala> print(p.isInstanceOf[Person])
true
scala>
3.getClass 和 classOf
isInstanceOf只能判断出对象是否是指定类以及其子类的对象,而不能精确判断出,对象就是指定类的对象
如果要求精确地判断对象就是指定类的对象,那么就只能使用getClass和classOf了
对象.getClass可以精确获取对象的类,classOf[类]可以精确获取类,然后使用==操作符即可判断
class Personclass Student extends Personval p: Person = new Studentp.isInstanceOf[Person]p.getClass == classOf[Person]p.isInstanceOf[Student]p.getClass == classOf[Student]
eg:
scala> p.getClass == classOf[Person]
res20: Boolean = false
scala> p.getClass == classOf[Student]
res21: Boolean = true
4.使用匹配模式来进行类型判断
但是在实际开发中,比如spark的源码中,大量的地方都是使用了模式匹配的方式来进行类型的判断,这种方式更加地简洁明了,而且代码得可维护性和可扩展性也非常的高
使用模式匹配,功能性上来说,与isInstanceOf一样,也是判断主要是该类以及该类的子类的对象即可,不是精准判断的
class Personclass Student extends Personval p: Person = new Studentp match {case per: Person => println("it's Person's object")case _ => println("unknown type")}
3.protected
跟java一样,scala中同样可以使用protected关键字来修饰field和method,这样在子类中就不需要super关键字,直接就可以访问非private修饰的field和method(非private修饰的field可以直接访问)
还可以使用protected[this],则只能在当前子类对象中访问父类的field和method,无法通过其他子类对象访问父类的field和method
class Person {protected var name: String = "leo"protected[this] var hobby: String = "game"}class Student extends Person {def sayHello = println("Hello, " + name)def makeFriends(s: Student) { //this s is not be current instance objectprintln("my hobby is " + hobby + ", your hobby is " + s.hobby)// error}}
eg1: error
scala> class Person {
| protected var name: String = "leo"
| protected[this] var hobby: String = "game"
| }
defined class Person
scala> class Student extends Person {
| def sayHello = println("Hello, " + name)
| def makeFriends(s: Student) {
| println("my hobby is " + hobby + ", your hobby is " + s.hobby)
| }
| }
<console>:15: error: value hobby is not a member of Student
println("my hobby is " + hobby + ", your hobby is " + s.hobby)
eg2:
scala> class Person {
| protected var name: String = "leo"
| protected var hobby:String = "game"
| }
defined class Person
scala> class Student extends Person {
| def sayHello = println("Hello, " + name)
| def makeFriends(s: Student) { //this s is not be current instance object
| println("my hobby is " + hobby + ", your hobby is " + s.hobby)
| }
| }
defined class Student
4.调用父类的constructor
Scala中,每个类可以有一个主constructor和任意多个辅助constructor,而每个辅助constructor的第一行都必须是调用其他辅助constructor或者是主constructor;因此子类的辅助constructor是一定不可能直接调用父类的constructor的
只能在子类的主constructor中调用父类的constructor,以下这种语法,就是通过子类的主构造函数来调用父类的构造函数
注意!如果是父类中接收的参数,比如name和age,子类中接收时,就不要用任何val或var来修饰了,否则会认为是子类要覆盖父类的field
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)}def this(age: Int) {this("leo", age, 0)}}
5.匿名内部类
在Scala中,匿名子类是非常常见,而且非常强大的。Spark的源码中也大量使用了这种匿名子类。
匿名子类,也就是说,可以定义一个类的没有名称的子类,并直接创建其对象,然后将对象的引用赋予一个变量。之后甚至可以将该匿名子类的对象传递给其他函数。
class Person(protected val name: String) {def sayHello = "Hello, I'm " + name}val p = new Person("leo") {//p is a anonymity child class's objectoverride def sayHello = "Hi, I'm " + name}def greeting(p: Person { def sayHello: String }) {println(p.sayHello)}greeting(p)//use anonymity class's object as a method parameter
6.抽象类 abstract class and abstract field
如果在父类中,有某些方法无法立即实现,而需要依赖不同的子类来覆盖,重写实现自己不同的方法实现。此时可以将父类中的这些方法不给出具体的实现,只有方法签名,这种方法就是抽象方法。
而一个类中如果有一个抽象方法,那么类就必须用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,以定义自己的具体field,并且覆盖抽象field,不需要使用override关键字
abstract class Person {val name: String}class Student extends Person {val name: String = "leo"}
scala 面向对象之 继承的更多相关文章
- Spark 3000门徒第二课scala面向对象总结
昨晚听了王家林老师3000门徒spark系列课程的第二课,讲述了scala面向对象知识,并且带着过了一遍Spark核心类:SparkContent,RDD的代码,下面写一下心得: RDD是抽象类,实现 ...
- Scala面向对象编程与类型系统
Scala支持面向对象编程, 其面向对象特性与Java有共同之处并添加了很多新的特性. 类定义 scala使用class关键字定义类: class MyComplex(real0:Double, im ...
- 黑马程序员_Java面向对象2_继承
4.面向对象_继承 4.1继承的概述 提高了代码的复用性. 让类与类之间产生了关系,有了这个关系,才有多态的特性. 注意:千万不要为了获取其他类的功能而去继承,简化代码而继承.必须是类与类之间有所属关 ...
- C++ Primer 学习笔记_69_面向对象编程 --继承情况下的类作用域
面向对象编程 --继承情况下的类作用域 引言: 在继承情况下,派生类的作用域嵌套在基类作用域中:假设不能在派生类作用域中确定名字,就在外围基类作用域中查找该名字的定义. 正是这样的类作用域的层次嵌套使 ...
- 【Java基础】【08面向对象_继承&方法&final】
08.01_面向对象(代码块的概述和分类)(了解)(面试的时候会问,开发不用或者很少用) A:代码块概述 在Java中,使用{}括起来的代码被称为代码块. B:代码块分类 根据其位置和声明的不同,可以 ...
- php面向对象 封装继承多态 接口、重载、抽象类、最终类总结
1.面向对象 封装继承多态 接口.重载.抽象类.最终类 面向对象 封装继承多态 首先,在解释面向对象之前先解释下什么是面向对象? [面向对象]1.什么是类? 具有相同属性(特征)和方法(行为)的一 ...
- 面向对象多继承(c3算法)、网络基础和编写网络相关的程序
一.面向对象多继承(c3算法) a.有多个父类先找左,再找右,如下示例: class A(object): pass class B(object): def f1(self): print('B') ...
- Scala的类继承
Scala的类继承 extend Scala扩展类的方式和java一样使用extends关键字 class Employee extends Person { } 与java一样,可以在定义的子类重写 ...
- Python 面向对象编程 继承 和多态
Python 面向对象编程 继承 和多态 一:多继承性 对于java我们熟悉的是一个类只能继承一个父类:但是对于C++ 一个子类可以有多个父亲,同样对于 Python一个类也可以有多个父亲 格式: c ...
随机推荐
- PHP学习之验证码类
<?php $code = new Code(); $code->outImage(); class Code { //验证码个数 protected $number; //验证码类型 p ...
- springboot 获取控制器参数的几种方式
这里介绍springboot 获取控制器参数有四种方式 1.无注解下获取参数 2.使用@RequestParam获取参数 3.传递数组 4.通过URL传递参数 无注解下获取参数无注解下获取参数,需要控 ...
- Oracle 中的进制转换
Oracle 中的进制转换 */--> Oracle 中的进制转换 Table of Contents 1. 进制名 2. 10进制与16进制互相转换 2.1. 10进制转换为16进制 2.2. ...
- 短信的内容提供者Uri和短信表结构
* sms表 * address :手机号码 * date :收发短信的时间 * read :短信的阅读状态 1,已读 0,未读 * type :收发短信的类型 1,收到短信 2,发出短信 * bod ...
- handler定义
Handler主要接收子线程发送的数据, 并用此数据配合主线程更新UI,用来跟UI主线程交互用.比如可以用handler发送一个message,然后在handler的线程中来接收.处理该消息,以避免直 ...
- 保存图片控件上的图片到本地 出现错误:无法将类型为“System.Windows.Media.Imaging.BitmapFrameDecode”的对象强制转换为类型“System.Windows.Media.Imaging.BitmapImage”。
保存图片控件上的图片到本地 出现错误:无法将类型为“System.Windows.Media.Imaging.BitmapFrameDecode”的对象强制转换为类型“System.Windows.M ...
- Java面试题准备(掌握)
1.使用什么方式进行文件上传? 一般使用FormData对象,ajax异步上传. 为什么使用? 1.FormData对象Mozilla上的介绍的,XMLHttpRequest Level 2添加了一个 ...
- Python中webbrowser的用法
#coding:utf-8 import time import webbrowser as web import os import random #随机选择一个浏览器打开网页 def open_u ...
- JS触发事件集锦
事件句柄 HTML 4.0 的新特性之一是有能力使 HTML 事件触发浏览器中的动作(action),比如当用户点击某个 HTML 元素时启动一段 JavaScript.下面是一个属性列表,这些属性可 ...
- Linux下查看文件编码及批量修改编码
查看文件编码在Linux中查看文件编码可以通过以下几种方式:1.在Vim中可以直接查看文件编码:set fileencoding即可显示文件编码格式.如果你只是想查看其它编码格式的文件或者想解决用Vi ...