Scala第七天——面向对象编程——继承

Scala基础 专栏收录该内容
13 篇文章 0 订阅

本文部分参考自:https://blog.csdn.net/dataiyangu/article/details/98483033


Scala第七天——面向对象编程

自己的话:漆黑的黑夜 表示着威胁 我选择诙谐

工作在堆叠 没兑现归结于那些理解

Scala继承之继承类、超类、匿名内部类、抽象类


一、extends(继承类)

1.简述:

(1)Scala中,让子类继承父类,与Java一样,也是使用extends关键字
(2)继承就代表,子类可以从父类继承父类的field和method;然后子类可以在自己内部放入父类所没有,子类特有的field和method;使用继承可以有效复用代码
(3)子类可以覆盖父类的field和method;但是如果父类用final修饰,field和method用final修饰,则该类是无法被继承的,field和method是无法被覆盖的

class Person {
  private var name = "leo"
  def getName = name
}
class Student extends Person {
  private var score = "A"
  def getScore = score
}

2.详解:

和 Java 一样使用 extends 关键字,在定义中给出子类需要而超类没有的 字段和方法,或者重写超类的方法。

class
Person {
  var name = ""
}
class Employee extends Person {
var salary = 0.0
def description = "An employee with name " + name + " and salary " + salary
}

如果类声明为 final,他讲不能被继承。如果单个方法声明为 final,将不 能被重写。
在这里插入图片描述
在这里插入图片描述


二、override、super(重写方法)

1.简述:

(1)Scala中,如果子类要覆盖一个父类中的非抽象方法,则必须使用override关键字
(2)override关键字可以帮助我们尽早地发现代码里的错误,比如:override修饰的父类方法的方法名我们拼写错了;比如要覆盖的父类方法的参数我们写错了;等等
(3)此外,在子类覆盖父类方法之后,如果我们在子类中就是要调用父类的被覆盖的方法呢?那就可以使用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 studetn my name is" + super.getName
}

2.详解:

重写一个非抽象方法需要用 override 修饰符。调用超类的方法使用 super 关键字

class
Person {
  var name = ""
override def toString = getClass.getName + "[name=" + name + "]" }
class Employee extends Person {
var salary = 0.0
override def toString = super.toString + "[salary=" + salary + "]" }

在这里插入图片描述

3.override field

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
}

三、isInstanceOf和asInstanceOf(类型检查和转换)

1.简述:

(1)如果我们创建了子类的对象,但是又将其赋予了父类类型的变量。则在后续的程序中,我们又需要将父类类型的变量转换为子类类型的变量,应该如何做?
(2)首先,需要使用isInstanceOf判断对象是否是指定类的对象,如果是的话,则可以使用asInstanceOf将对象转换为指定类型
注意,如果对象是null,则isInstanceOf一定返回false,asInstanceOf一定返回null

注意,如果没有用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]

2.详解:

要测试某个对象是否属于某个给定的类,可以用 isInstanceOf 方法。用asInstanceOf 方法将引用转换为子类的引用。classof 获取对象的类名。
在这里插入图片描述
在这里插入图片描述


四、getClass和classOf

1、isInstanceOf只能判断出对象是否是指定类以及其子类的对象,而不能精确判断出,对象就是指定类的对象

2、如果要求精确地判断对象就是指定类的对象,那么就只能使用getClass和classOf了

3、对象.getClass可以精确获取对象的类,classOf[类]可以精确获取类,然后使用==操作符即可判断

class Person
class Student extends Person
val p: Person = new Student
p.isInstanceOf[Person]
p.getClass == classOf[Person]
p.getClass == classOf[Student]

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

1、但是在实际开发中,比如spark的源码中,大量的地方都是使用了模式匹配的方式来进行类型的判断,这种方式更加地简洁明了,而且代码得可维护性和可扩展性也非常的高

2、使用模式匹配,功能性上来说,与isInstanceOf一样,也是判断主要是该类以及该类的子类的对象即可,不是精准判断的

class Person
class Student extends Person
val p: Person = new Student

p match {
  case per: Person => println("it's Person's object")
  case _  => println("unknown type")
}

六、protected(受保护的字段和方法)

1.简述:

(1)跟java一样,scala中同样可以使用protected关键字来修饰field和method,这样在子类中就不需要super关键字,直接就可以访问field和method

(2)还可以使用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) {
    println("my hobby is " + hobby + ", your hobby is " + s.hobby)
  }
}

2.详解:

protected 声明字段和方法,子类可见,所属包不可见。
在这里插入图片描述
在这里插入图片描述
上述例子中的类和子类的salary这个字段是protected,所以是能过相互访问的。


七、超类的构造(调用父类的constructor)

1.简述:

(1)Scala中,每个类可以有一个主constructor和任意多个辅助constructor,而每个辅助constructor的第一行都必须是调用其他辅助constructor或者是主constructor;因此子类的辅助constructor是一定不可能直接调用父类的constructor的

(2)只能在子类的主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)
  }
}

2.详解:

只有主构造器才能调用超类的构造器

class
Person(val name: String, val age: Int) {
override def toString = getClass.getName + "[name=" + name +
    ",age=" + age + "]"
}

class Employee(name: String, age: Int, val salary : Double) extends Person(name, age) {
 override def toString = super.toString + "[salary=" + salary + "]" }

上面的第一句话中的主构造器的意思是(name: String, age: Int, val salary : Double)

上面的extends后面的name age都会放到Employee前面的构造中的name age里面,通过这种方式子类去访问父类的构造器

构造器的参数,传入了超类的构造函数。

scala中不需要像java一样super调用构造器,只需要extends(),就可以了。

Scala 不能通过 super(a)这种方式调用超类的构造函数。

在这里插入图片描述


八、重写字段

子类改写父类或者抽象父类的字段,通过以下方式:

abstract class Person { // See Section 8.8 for abstract classes def id: Int // Each person has an ID that is computed in some way
}
//第一种方法,直接在构造器里面override val id: Int
class Student(override val id: Int) extends Person

//第二种方法在类里面 
class SecretAgent extends Person {
override val id = scala.util.Random.nextInt
}

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
把aaa删掉
在这里插入图片描述


九、匿名内部类(匿名子类)

1.简述:

(1)在Scala中,匿名子类是非常常见,而且非常强大的。

(2)匿名子类,也就是说,可以定义一个类的没有名称的子类,并直接创建其对象,然后将对象的引用赋予一个变量。之后甚至可以将该匿名子类的对象传递给其他函数。

class Person(protected val name: String) {
  def sayHello = "Hello, I'm " + name
}
在object对象中写
val p = new Person("leo") {
  override def sayHello = "Hi, I'm " + name
}
定义一个方法,接受的方法参数就是Person类还有一个sayHello方法,和上边定义的匿名内部类结构一样
def greeting(p: Person { def sayHello: String }) {
  println(p.sayHello)
}
greeting(p)

2.详解:

和 Java 一样,你可以通过包含带有定义或重写的代码块的方式创建一个

匿名的子类:

class Person(val name: String) {
override def toString = getClass.getName + "[name=" + name + "]"
}
 //希望声明一个匿名子类
//new了一个父类
val alien = new Person("Fred") {	
//提供了子类的一些方法,比如新加了一个greeting方法 
//这个时候就已经有了子类了,但是这个子类其实是一个匿名子类
def greeting = "Greetings, Earthling! My name is Fred."
}

匿名类的变量类型为:Person{def greeting: String} 这是一个结构类型
在这里插入图片描述
如上图,匿名meet方法里面的参数是一个匿名子类,红圈的部分是匿名子类的类型

这里注意
在这里插入图片描述
把上面的abc删掉后,这里就会报错
在这里插入图片描述
但是如果把下面的类型中的后面去掉,是可以运行成功的,只是有一个warning,因为只要能满足我下面参数中的类型就行了,虽然上面的类是一个匿名方法,但是如果能满足下面的东西的类型需求的就可以


十、抽象类

1.简述:

(1)如果在父类中,有某些方法无法立即实现,而需要依赖不同的子来来覆盖,重写实现自己不同的方法实现。此时可以将父类中的这些方法不给出具体的实现,只有方法签名,这种方法就是抽象方法。

(2)而一个类中如果有一个抽象方法(可以有具体的方法),那么类就必须用abstract来声明为抽象类,此时抽象类是不可以实例化的
抽象类不一定有抽象方法,有抽象方法的类一定第抽象类。

(3)在子类中覆盖抽象类的抽象方法时,不需要使用override关键字

abstract class Person(val name: String) {
  def sayHello: Unit
}
class Student(name: String) extends Person(name) {
  def sayHello: Unit = println("Hello, " + name)
}

2.详解:

可以通过 abstract 关键字标记不能被实例化的类。方法不用标记 abstract,只要省掉方法体即可。类可以拥有抽象字段,抽象字段就是没有初 始值的字段。

abstract class Person(val pname: String) {
val id: Int
// No initializer—this is an abstract field with an abstract getter method var name: String
// Another abstract field, with abstract getter and setter methods
def idString: Int // No method body—this is an abstract method
}
class Employee(pname: String) extends Person(pname) {
val id = 5;
var name = ">>>"
def idString = pname.hashCode // override keyword not required
}

一旦继承了抽象类,就要复写所有的抽象字段和方法,如果不写的话,编译器也会主动要求进行复写

子类实现抽象方法不需要 override。
override是复写非抽象方法的时候用的

在这里插入图片描述

3.抽象field

(1)如果在父类中,定义了field,但是没有给出初始值,则此field为抽象field
(2)抽象field意味着,scala会根据自己的规则,为var或val类型的field生成对应的getter和setter方法,但是父类中是没有该field的
(3)子类必须覆盖field,以定义自己的具体field,并且覆盖抽象field,不需要使用override关键字

abstract class Person {
  val name: String
}

class Student extends Person {
  val name: String = "leo"
}

十一、Scala 继承层级

在这里插入图片描述
在这里插入图片描述


让优秀成为习惯


  • 0
    点赞
  • 1
    评论
  • 0
    收藏
  • 一键三连
    一键三连
  • 扫一扫,分享海报

©️2021 CSDN 皮肤主题: 游动-白 设计师:白松林 返回首页
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、C币套餐、付费专栏及课程。

余额充值