Scala 【 8 面向对象编程 - 继承 】

2023-03-09 14:41:51 浏览数 (1)

面向对象编程 - 继承

复用 ~ 可扩展性 ~ 可维护性

extends

使用 extends 关键字表示继承。

继承就代表,子类可以从父类继承父类的 field 和 method 。子类可以在自己内部放入父类所没有的子类特有的 field 和 method 。

使用继承可以有效复用代码。

子类可以覆盖父类的 field 和 method 。

如果父类用 final 修饰, field 和 method 用 final 修饰,则该类无法被继承的, field 和 method 是无法被覆盖的。

override 和 super

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

override 关键字可以帮助我们尽早地发现代码里的错误。

在子类覆盖父类方法之后,想要调用父类中被覆盖的方法则用 super 关键字。

代码语言:javascript复制
class Person{
	private var name = "Li"
	def getName = name
}

class Student extends Person{
	private var score = "A"
	def getScore = score
	overrider def getName = "Hi,"   super.getName
}
override field

子类可以覆盖父类的 val field ,而且子类的 val field 还可以覆盖父类的 val field 的 getter 方法。使用 override 关键子。

代码语言:javascript复制
class Person{
	val name:String = "Person"
	def age:Int = 0
}

class Student extends Person{
	override val name:String = "Li"
	override val age:Int = 19
}
isInstanceOf 和 asInstanceOf

如果我们创建了子类的对象,但是又将其赋予了父类类型的变量。在后续的程序中,又需要将父类类型的变量转换为子类类型的变量,如何改变?

首先,需要使用 inInstanceOf 判断对象是否是指定类的对象,如果是的话,则可以使用 asInstanceOf 将对象转换为指定类型。

**注意:**如果对象是 null ,则 isInstanceOf 一定返回 false ,asInstanceOf 一定返回 null 。

**注意:**如果没有用 isInstanceOf 先判断对象是否为指定类的实例,就直接用 asInstanceOf 转换,则可能会抛出异常。

父类转变成子类,用 isInstaceOf

和 asInstanceOf 。

代码语言:javascript复制
class Person
class Student extends Person
val p:Person = new Student
var s:Student = null
if(p.isInstaceOf[Student])s=p.asInstanceOf[Student]
getClass 和 classOf

isInstanceOf 只能判断出对象是否是指定类以及其之类的对象,不能精确判断出对象就是指定类的对象。

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

classOf 了。

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

代码语言:javascript复制
class Person
class Student extends Person
val p:Person = new Student
p.isInstanceOf[Person]
p.getClass
// p.getClass 的返回: Class[_ <: Person] = class Student

p.getClass == classOf[Person]
// 返回为 flase

p.getClass == classOf[Student]
// 返回为 true
使用模式匹配进行类型判断

实际开发中,很多地方都使用模式匹配的方式来进行类型的判断,这种方式更加简洁明了,而且代码的可维护性和可扩展性也非常高。

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

代码语言:javascript复制
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

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

还可以使用 protected[this],这样只能在当前子类对象中访问父类的 field 和 method ,无法通过其他子类对象访问父类的 field 和 method 。

代码语言:javascript复制
// example1
class Person{
    protected var name:String = "Li"
}
class Student extends Person{
    def makeFriends(s:Student){
        println("Hi, my name is"   name   ", your name is "   s.name)
    }
}

val s1 = new Student
val s2 = new Student
s1.makeFriends(s2)
// 这里输出为:
// Hi,my name is li,your name is li
// 所以这里不仅可以通过继承来访问父类的 field ,还可以通过其他实例来访问。
代码语言:javascript复制
// example2
// 会报错
class Person{
    protected[this] val name:String = "Li"
}
class Student extends Person{
    def makeFriends(s:Student){
         println("Hi, my name is"   name   ", your name is "   s.name)
    }
}

以上定义会报错:

代码语言:javascript复制
<pastie>:18: error: value name is not a member of Student
                println("Hi, my name is"   name   ", your name is "   s.name)
调用父类的 constructor

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

只能在子类的主 constructor 中调用父类的 constructor 。

如果是父类中接收的参数,比如 name 和 age,子类中接受时,就不要用任何 val 或者 var 来修饰了,否则认为是子类要覆盖父类的 field 。

代码语言:javascript复制
class Person(val name:String, val age:Int)

// 这里的 name 和 age 就是使用的父类的。
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("Li",age,0)
    }
}

val s = new Student("Li",20,100)
匿名内部类

定义一个类的没有名称的子类,并直接创建其对象,然后将对象的引用赋予一个变量。还可以将该匿名子类的对象传递给其他函数。

代码语言:javascript复制
class Person(val name:String){
    def sayHello = "Hello,I'm "   name
}

// 以下就是匿名内部类
val p = new Person("Li"){
    override def sayHello = "Hi,I'm "   name 
}

def greeting(p: Person{def sayHello:String}){
    println(p.sayHello)
}
抽象类

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

在类中如果有一个抽象方法,那么类就必须用abstract 来声明为抽象类,此时抽象类是不可以实例化的。

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

代码语言:javascript复制
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 。

抽象 field 意味着,scala 会根据自己的规则,为 var 或 val 类型的 field 生成对应的 getter 和 setter 方法,但是父类中是没有该 field 的。

子类必须覆盖 field 来定义自己具体的 field ,并且覆盖抽象 field ,不需要使用 override 关键字。

代码语言:javascript复制
abstract class Person{
	val name:String
}

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

0 人点赞