scala面向对象之继承

extends

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

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

override和super

scala中如果子类要覆盖一个父类中的抽象方法,则必须使用override关键字
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 studetn my name is" + super.getName
}

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
}

inlnstanceOf和aslnstanceOf

如果我们创建子类对象,但是又将其赋予了父类类型的变量,则在后续的程序中,我们又需要将父类类型的变量转换为子类类型的变量应该怎么做呢?
首先需要使用inlnstanceOf判断对象是否是指定类的对象,如果是的话则可以使用iaslnstanceOf
将对象转化为指定的类型
注意:
(1)如果对象是null,则inlnstanceOf一定返回false,aslnstanceOf一定返回null
(2)如果没有用inlnstanceOf先判断对象是否为指定类的实例,就直接用aslnstanceOf装换则可能会抛出异常

class Person
class Student extends Person
val p: Person =  new Student
var s: Student = null
if (p.isInstanceOf[Student]) s = p.asInstanceOf[Student]

getClass和classOf

isInstanceOf只能判断出对象是否是指定类以及子类的对象,而不能判断出对象就是指定类的对象
如果要精确判断对象就是指定对象,那么就只能使用getClass和classOf
对象.getClass可以精确获取对象的类,classOf[类]可以精确获取类,然后使用==操作符即可判断

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

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

在实际开发中,大量的地方都是用模式匹配的方式来进行类型的判断,这种方式更加简洁明了,而且代码可维护性,可扩展性也非常高
使用模式匹配功能上来说,与inlnstanceOf一样,可以主要判断是该类以及该类子类的对象即可,不是精确判断的

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

与Java一样,scala中同样可以使用protected关键字来修饰field和method,这样在子类中就不需要super关键字就可以直接访问field和method
还可以使用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)
  }
}

调用父类的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)
  }
}

匿名内部类

在scala中,匿名子类是非常常见,而且非常强大的
匿名子类也就是说可以定义一个类的没有名称的子类,把那个直接创建期对象,然后将对象的引用赋予一个变量,之后甚至可以将该匿名子类的对象传递给其他函数

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)

抽象类

如果在父类中,有某些方法无法立即实现,而需要依赖不同的子类来覆盖,重写实现自己不同的方法,此时可以将父类中的这些方法不给出具体的实现,只有方法签名,这种方法就是抽象方法
一个类中如果有一个抽象方法(可以有具体的方法),那么方法必须用obstract来声明为抽象类,此时抽象类是不可以实例化的
在子类中覆盖抽象方法时,不需要使用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
抽象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"
}
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值