scala笔记第六天(继承)

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接: https://blog.csdn.net/TylerPY/article/details/102668388

extends

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

class Persion_12_1 {
  private var name = "leo"
  def getName = name

}

class Student_12_2 extends Persion_12_1 {
  private var score = "A"
  def getScore = score
}

object TextDemo {
  def main(args: Array[String]): Unit = {
    val a = new Student_12_2
    println(a.getScore)  //A
    println(a.getName)   //leo
  }
}

override和super

Scala中,如果子类要覆盖一个父类中的非抽象方法,则必须使用override关键字。override关键字可以帮助我们尽早地发现代码里的错误,比如:override修饰的父类方法的方法名我们拼写错了;比如要覆盖的父类方法的参数我们写错了;此外,在子类覆盖父类方法之后,如果我们在子类中就是要调用父类的被覆盖的方法呢?那就可以使用super关键字,显式地指定要调用父类的方法。

class persion_12_2 {
  private var name = "tom"
  def getName = name
}

class Students_12_2 extends persion_12_2 {
  private var score = "A"
  def getScore = score

  override def getName: String = "Hi, I'm studetn my name is "+ super.getName


}


object TextDemo {
  def main(args: Array[String]): Unit = {
    //12 ----------2---------override和super
    val a = new Students_12_2
    println(a.getName)  //Hi, I'm studetn my name is tom
    println(a.getScore)   //A
  }
}

override field(字段)

Scala中,子类可以覆盖父类的val field,而且子类的val field还可以覆盖父类的val field的getter方法;只要在子类中使用override关键字即可。

class Persion12_3 {
  val name:String = "persion"
  def age:Int = 0
}

class Student_12_3 extends Persion12_3 {
  override def age: Int = 30

  override val name: String = "leo"
}

object TextDemo {
  def main(args: Array[String]): Unit = {
    //12 ----------3---------override field(字段)
    val a = new Student_12_3
    println(a.name)  //leo
    println(a.age)  //30
  }
}

isInstanceOf和asInstanceOf

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

class Perison_12_4 {
  val name = "Persion"
}

class Student_12_4 extends Perison_12_4 {
  val namea = "wang"

}

object TextDemo {
  def main(args: Array[String]): Unit = {
    //12 ----------4--------- isInstanceOf和asInstanceOf
    val p :Perison_12_4 = new Student_12_4
    println(p.name)
    var s : Student_12_4 = null
    println(p.isInstanceOf[Student_12_4])
    s = p.asInstanceOf[Student_12_4]
    println(s.namea)
  }
}

getClass和classOf

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

class Persion_12_5 {

}

class Student_12_5 extends Persion_12_5 {

}

object TextDemo {
  def main(args: Array[String]): Unit = {
    //12 ----------5--------- getClass和classOf
    val p :Persion_12_5 = new Student_12_5
    println(p.isInstanceOf[Persion_12_5])  //true
    println(p.getClass == classOf[Persion_12_5])   //false
    println(p.getClass == classOf[Student_12_5])    //true
  }
}

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

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

class Persion_12_6 {

}

class Student_12_6  {

}


object TextDemo {
  def main(args: Array[String]): Unit = {
    //12 ----------6---------  使用模式匹配进行类型判断
    //val p = new Persion_12_6   //Persion object
    //val p = new Student_12_6   //Student object
    val p = new Persion_12_5    //unkown type
    p match {
      case per:Persion_12_6 => println("Persion object")
      case stu :Student_12_6 => println("Student object")
      case _ => println("unkown type")
    }
  }
}

protected

跟java一样,scala中同样可以使用protected关键字来修饰field(字段)和method(方法),这样在子类中就不需要super关键字,直接就可以访问field和method。还可以使用protected[this],则只能在当前子类对象中访问父类的field和method,无法通过其他子类对象访问父类的field和method。

scala对private和private[this]做的工作:

(1)对于类私有的字段,Scala生成私有的getter/setter方法;

(2)对于对象私有的字段,Scala不生成getter/setter方法。

class Persion_12_7 {
  protected var name : String = "leo"
  protected [this] var hobby: String = "game"

}

class Students_12_7 extends Persion_12_7 {
  def sayHello = print("Hello" + name)
//  def makeFriends(s:Students_12_7): Unit ={
//    println("my hobby is " + hobby + ", your hobby is " + s.hobby)
//  }
}


object TextDemo {
  def main(args: Array[String]): Unit = {
    //12 ----------7---------protected
    val a = new Students_12_7
    a.sayHello   //Helloleo
  }
}

调用父类的constructor

Scala中,每个类可以有一个主constructor和任意多个辅助constructor,而每个辅助constructor的第一行都必须是调用其他辅助constructor或者是主constructor;因此子类的辅助constructor是一定不可能直接调用父类的constructor的。只能在子类的主constructor中调用父类的constructor,以下这种语法,就是通过子类的主构造函数来调用父类的构造函数。注意!如果是父类中接收的参数,比如name和age,子类中接收时,就不要用任何val或var来修饰了,否则会认为是子类要覆盖父类的field。

扫描二维码关注公众号,回复: 7573603 查看本文章
class Persion_12_8(val name:String,val age:Int){

}

class Student_12_8(name: String, age: Int, var score: Double) extends Persion_12_8(name, age) {
  def this(name:String){
    this(name,0,0)
  }
  def this(age:Int){
    this("leo",age,0)
  }
  def this(age:Int,score:Double){
    this("a",age,score)
  }
}

object TextDemo {
  def main(args: Array[String]): Unit = {
    //12 ----------8---------调用父类的constructor
    val s1 = new Student_12_8("wang",10,2.3)
    val s2 = new Student_12_8("liu")
    val s3 = new Student_12_8(34)
    val s4 = new Student_12_8(23,23.7)
  }
}

匿名内部类

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

class Persion_12_9(protected val name:String){
  def sayHello = " Hello i am " + name
}


object TextDemo {
  def main(args: Array[String]): Unit = {
    //12 ----------9---------匿名内部类
    val p = new Persion_12_9("tom"){
      override def sayHello: String = "haha "  + name
    }
    //println(p.sayHello)   //haha tom
    def greeting(p: Persion_12_9 { def sayHello: String }) {
      println(p.sayHello)
    }
    greeting(p)   //haha tom


  }
}

抽象类

如果在父类中,有某些方法无法立即实现,而需要依赖不同的子来来覆盖,重写实现自己不同的方法实现。此时可以将父类中的这些方法不给出具体的实现,只有方法签名,这种方法就是抽象方法。而一个类中如果有一个抽象方法(可以有具体的方法),那么类就必须用abstract来声明为抽象类,此时抽象类是不可以实例化的
抽象类不一定有抽象方法,有抽象方法的类一定第抽象类。 在子类中覆盖抽象类的抽象方法时,不需要使用override关键字。

abstract class Persion_12_10(val name:String) {
  def sayHello:Unit
  def sayHi()= {
    println("Hi " + name)
  }
}

class Student_12_10(name: String) extends Persion_12_10(name ) {
  def sayHello: Unit = {
    println("Hello ," + name)
  }
}

object TextDemo {
  def main(args: Array[String]): Unit = {
    //12 ----------10---------抽象类
    val s :Persion_12_10= new Student_12_10("wang")
    val s1 = new Student_12_10("wang")
    s1.sayHello   //Hello ,wang
    s.sayHello    //Hello ,wang
    s.sayHi()     //Hi wang
    s1.sayHi()    //Hi wang
  }
}

抽象field(字段)

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

abstract class Persion_12_11 {
  val name:String
}

class Student_12_11 extends Persion_12_11 {
  val name : String = "leo"

}


object TextDemo {
  def main(args: Array[String]): Unit = {
    //12 ----------11---------抽象field(字段)
    val a = new Persion_12_11 {
      override val name: String = "tom"
    }
    println(a.name)  //tom
    val s = new Student_12_11
    println(s.name)  //leo

  }
}

猜你喜欢

转载自blog.csdn.net/TylerPY/article/details/102668388