七、scala面向对象-继承

1、继承java

一、extendses6

Scala中,让子类继承父类,与Java同样,也是使用extends关键字 
继承就表明,子类能够从父类继承父类的field和method;而后子类能够在本身内部放入父类所没有,
子类特有的field和method;使用继承能够有效复用代码 
子类能够覆盖父类的field和method;可是若是父类用final修饰,field和method用final修饰,则该类是没法被继承的,
field和method是没法被覆盖的;


###
scala> :paste
// Entering paste mode (ctrl-D to finish)

class Person {
  private var name = "leo"
  def getName = name
}

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

// Exiting paste mode, now interpreting.

defined class Person
defined class Student
warning: previously defined object Person is not a companion to class Person.
Companions must be defined together; you may wish to use :paste mode for this.

scala> val s = new Student
s: Student = Student@985696

scala> s.getName
res11: String = leo

scala> s.getScore
res12: String = A

scala> s.getName
res13: String = leo

scala> s.name
<console>:14: error: value name is not a member of Student
       s.name
         ^


二、override和superide

//Scala中,若是子类要覆盖一个父类中的非抽象方法,则必须使用override关键字;

//override关键字能够帮助咱们尽早地发现代码里的错误,好比:override修饰的父类方法的方法名咱们拼写错了;好比要覆盖的父类方法的参数咱们写错了;等等

//此外,在子类覆盖父类方法以后,若是咱们在子类中就是要调用父类的被覆盖的方法呢?
   那就可使用super关键字,显式地指定要调用父类的方法


###
scala> :paste
// Entering paste mode (ctrl-D to finish)

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 a student, my name is " + super.getName
}

// Exiting paste mode, now interpreting.

defined class Person
defined class Student
warning: previously defined object Person is not a companion to class Person.
Companions must be defined together; you may wish to use :paste mode for this.

scala> val s = new Student
s: Student = Student@618ad2aa

scala> s.getName
res15: String = Hi, I'm a student, my name is leo


三、override field函数

// Scala中,子类能够覆盖父类的val field,并且子类的val field还能够覆盖父类的val field的getter方法;只要在子类中使用override关键字便可

###
scala> :paste
// Entering paste mode (ctrl-D to finish)

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

class Student extends Person { 
  override val name: String  = "leo"
  override val age: Int = 30
}

// Exiting paste mode, now interpreting.

defined class Person
defined class Student
warning: previously defined object Person is not a companion to class Person.
Companions must be defined together; you may wish to use :paste mode for this.

scala> val p = new Person
p: Person = Person@71d9cb05

scala> p.name
res16: String = Person

scala> p.age
res17: Int = 0

scala> val s = new Student
s: Student = Student@1ac4ccad

scala> s.name
res18: String = leo

scala> s.age
res19: Int = 30


四、isInstanceOf和asInstanceOfthis

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



###
scala> class Person
defined class Person
warning: previously defined object Person is not a companion to class Person.
Companions must be defined together; you may wish to use :paste mode for this.

scala> class Student extends Person
defined class Student

scala> val p: Person = new Student
p: Person = Student@f324455

scala> var s: Student = null
s: Student = null

scala> if (p.isInstanceOf[Student]) s = p.asInstanceOf[Student]


五、getClass和classOfspa

// isInstanceOf只能判断出对象是不是指定类以及其子类的对象,而不能精确判断出,对象就是指定类的对象
// 若是要求精确地判断对象就是指定类的对象,那么就只能使用getClass和classOf了
// 对象.getClass能够精确获取对象的类,classOf[类]能够精确获取类,而后使用==操做符便可判断


###
scala> class Person
defined class Person

scala> class Student extends Person
defined class Student

scala> val p: Person = new Student
p: Person = Student@77468bd9

scala> p.isInstanceOf[Person]
res0: Boolean = true

scala> p.getClass == classOf[Person]
res1: Boolean = false

scala> p.getClass == classOf[Student]
res2: Boolean = true


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

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



###
scala> class Person
defined class Person

scala> class Student extends Person
defined class Student

scala> val p: Person = new Student
p: Person = Student@192d3247

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


七、protectedcode

// 跟java同样,scala中一样可使用protected关键字来修饰field和method,这样在子类中就不须要super关键字,直接就能够访问field和method
// 还可使用protected[this],则只能在当前子类对象中访问父类的field和method,没法经过其余子类对象访问父类的field和method


###
scala> :paste
// Entering paste mode (ctrl-D to finish)

class Person {
  protected var name: String = "leo"
}

class Student extends Person {
  def makeFriends(s: Student) {
    println("Hi, my name is " + name + ", your name is " + s.name)
  }
}

// Exiting paste mode, now interpreting.

defined class Person
defined class Student

scala> val s1 = new Student
s1: Student = Student@63355449

scala> val s2 = new Student
s2: Student = Student@1ab3a8c8

scala> s1.makeFriends(s2)
Hi, my name is leo, your name is leo




scala> :paste
// Entering paste mode (ctrl-D to finish)

class Person {
  protected[this] val name: String  = "leo"
}

class Student extends Person {
  def makeFriends(s: Student) {
    println("my name is " + name + ", your name is " + s.name)
  }
}

// Exiting paste mode, now interpreting.

<console>:22: error: value name is not a member of Student
           println("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


###
scala> :paste
// Entering paste mode (ctrl-D to finish)

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.0)
  }
  def this(age: Int) {
    this("leo", age, 0)
  }
}

// Exiting paste mode, now interpreting.

defined class Person
defined class Student

scala> val s = new Student("leo", 30, 100)
s: Student = Student@783a467b

scala> s.name
res6: String = leo

scala> s.age
res7: Int = 30

scala> s.score
res8: Double = 100.0

scala> val s2 = new Student("leo")
s2: Student = Student@6f204a1a

scala> s2.name
res9: String = leo

scala> s2.age
res10: Int = 0

scala> val s3 = new Student(30)
s3: Student = Student@3224a577

scala> s3.name
res11: String = leo

scala> s3.age
res12: Int = 30

scala> s3.score
res13: Double = 0.0


九、匿名子类blog

// 在Scala中,匿名子类是很是常见,并且很是强大的。Spark的源码中也大量使用了这种匿名子类。
// 匿名子类,也就是说,能够定义一个类的没有名称的子类,并直接建立其对象,而后将对象的引用赋予一个变量。以后甚至能够将该匿名子类的对象传递给其余函数。


###
scala> :paste
// Entering paste mode (ctrl-D to finish)

class Person(protected val name: String) {
  def sayHello = "Hello, I'm " + name
}

val p = new Person("leo") {
  override def sayHello = "Hi, I'm " + name
}

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

// Exiting paste mode, now interpreting.

defined class Person
p: Person = $anon$1@3f270e0a
greeting: (p: Person{def sayHello: String})Unit

scala> greeting(p)
Hi, I'm leo


十、抽象类

// 若是在父类中,有某些方法没法当即实现,而须要依赖不一样的子来来覆盖,重写实现本身不一样的方法实现。此时能够将父类中的这些方法不给出具体的实现,
只有方法签名,这种方法就是抽象方法。
// 而一个类中若是有一个抽象方法,那么类就必须用abstract来声明为抽象类,此时抽象类是不能够实例化的
// 在子类中覆盖抽象类的抽象方法时,不须要使用override关键字



###
scala> :paste
// Entering paste mode (ctrl-D to finish)

abstract class Person(val name: String) {
  def sayHello: Unit
}

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

// Exiting paste mode, now interpreting.

defined class Person
defined class Student

scala> val p = new Person
<console>:11: error: class Person is abstract; cannot be instantiated
       val p = new Person
               ^

scala> val s = new Student("leo")
s: Student = Student@368239c8

scala> s.sayHello
Hello, leo


十一、抽象field

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


###
scala> :paste
// Entering paste mode (ctrl-D to finish)

abstract class Person {
  val name: String
}

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

// Exiting paste mode, now interpreting.

defined class Person
defined class Student

scala> val s = new Student
s: Student = Student@1794d431

scala> s.name
res1: String = leo
相关文章
相关标签/搜索