scala(4)面向对象编程之对象、继承

object 伴生对象

package com.doit.scc1

object Person {
  // object,相当于class的单个实例,通常在里面放一些静态的field或者method
  // 第一次调用object的方法时,就会执行object的constructor,也就是object内部不在method中的代码;但是object不能定义接受参数的constructor
  // 注意,object的constructor只会在其第一次被调用时执行一次,以后再次调用就不会再次执行constructor了
  // object通常用于作为单例模式的实现,或者放class的静态成员,比如工具方法
  private var eyeNum = 2
  println("this Person object!")
  def getEyeNum = eyeNum
}
class Person(val name:String,val age:Int) {
  def sayHello = println("Hi," + name + ",I guess you are " + age + " years old!" +
    ", and usually you must have " + Person.eyeNum + " eyes.")
}
测试:val s = new Person("ynn",23)
  	s.sayHello
运行结果:Hi,ynn,I guess you are 23 years old!, and usually you must have 2 eyes.

让object继承抽象类

package com.doit.scc1

// object的功能其实和class类似,除了不能定义接受参数的constructor之外
// object也可以继承抽象类,并覆盖抽象类中的方法
abstract class Hello(var message:String) {
  def sayHello(name:String):Unit
}
object Hellolmpl extends Hello("hello"){
  override def sayHello(name: String) = {
    println(message + ","+ name)
  }
}
测试: Hellolmpl.sayHello("ynn")
运行结果:hello,ynn

apply方法

package com.doit.scc1
// object中非常重要的一个特殊方法,就是apply方法
// 通常在伴生对象中实现apply方法,并在其中实现构造伴生类的对象的功能
// 而创建伴生类的对象时,通常不会使用new Class的方式,而是使用Class()的方式,隐式地调用伴生对象得apply方法,这样会让对象创建更加简洁
class Person_2(val name:String) {

}
object Person_2{
  def apply(name:String) = {new Person_2(name);print("hello" + name)}
}
测试:Person_2("ynn")
运行结果:helloynn

main方法

// 就如同java中,如果要运行一个程序,必须编写一个包含main方法类一样;在scala中,如果要运行一个应用程序,那么必须有一个main方法,作为入口
// scala中的main方法定义为def main(args: Array[String]),而且必须定义在object中
object HelloWorld {
  def main(args: Array[String]) {
    println("Hello World!!!")
  }
}
//在桌面创建一个HelloWorld.scala,内容如上,在windows上进入cmd,执行scalac HelloWorld.scala编译scala文件 ,产生一个HelloWorld.class,然后scala HelloWorld
C:\Users\96467>cd C:\Users\96467\Desktop

C:\Users\96467\Desktop>scalac HelloWorld.scala

C:\Users\96467\Desktop>scala HelloWorld
Hello World!!!

// 除了自己实现main方法之外,还可以继承App Trait,然后将需要在main方法中运行的代码,直接作为object的constructor代码;而且用args可以接受传入的参数

用object来实现枚举功能

package com.doit.scc1
// Scala没有直接提供类似于Java中的Enum这样的枚举特性,如果要实现枚举,则需要用object继承Enumeration类,并且调用Value方法来初始化枚举值
object Season extends Enumeration {
  var SPRING,SUMMER,AUTUMN,WINTER = Value
}
测试:for (e <- Season.values) println(e)  // 使用枚举object.values可以遍历枚举值
运行结果:
SPRING
SUMMER
AUTUMN
WINTER

// 还可以通过Value传入枚举值的id和name,通过id和toString可以获取; 还可以通过id和name来查找枚举值
object Season extends Enumeration {
  val SPRING = Value(0,"spring")
  val SUMMER = Value(1,"summer")
  val AUTUMN = Value(2,"autumn")
  val WINTER = Value(3,"winter")
}
测试:for (e <- Season.values) println(e)  // 使用枚举object.values可以遍历枚举值
运行结果:
spring
summer
autumn
winter

extends

class Person {
  private var name = "ynn"
  def getName = name
}
// Scala中,让子类继承父类,与Java一样,也是使用extends关键字
// 继承就代表,子类可以从父类继承父类的field和method;然后子类可以在自己内部放入父类所没有,子类特有的field和method;使用继承可以有效复用代码
// 子类可以覆盖父类的field和method;但是如果父类用final修饰,field和method用final修饰,则该类是无法被继承的,field和method是无法被覆盖的
class Student extends Person{
  private var score = "A"
  def getScore = score
}
测试:val s = new Student
  	println(s.getScore)
  	println(s.getName)
运行结果:
A
ynn

override和super

class Person {
  private var name = "ynn"
  def getName = name
}
class Student extends Person{
  private var score = "A"
  def getScore = score
  // Scala中,如果子类要覆盖一个父类中的非抽象方法,则必须使用override关键字
  override def getName: String = "Hi,I'm student,my name is "+super.getName
  // override关键字可以帮助我们尽早地发现代码里的错误,比如:override修饰的父类方法的方法名我们拼写错了;比如要覆盖的父类方法的参数我们写错了;等等
// 此外,在子类覆盖父类方法之后,如果我们在子类中就是要调用父类的被覆盖的方法呢?那就可以使用super关键字,显式地指定要调用父类的方法
}
测试:val s = new Student
  	println(s.getName)
运行结果:Hi,I'm student,my name is ynn

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
}
测试:  val s = new Student
	  println(s.name)
	  println(s.age)
运行结果:
leo
30

isInstanceOf 和 asInstanceOf

//使用isInstanceOf 判断对象是否为指定类的对象,如果是的话,则可以使用 asInstanceOf 将对象转换为指定类型
//注意:p.isInstanceOf[XX] 判断 p 是否为 XX 对象的实例;p.asInstanceOf[XX] 把 p 转换成 XX 对象的实例
//注意:如果没有用isInstanceOf 先判断对象是否为指定类的实例,就直接用 asInstanceOf 转换,则可能会抛出异常;
//注意:如果对象是 null,则isInstanceOf 一定返回 false, asInstanceOf 一定返回 null
class Person2 {}
class Student2  extends Person2{}
测试:
  val p : Person2 = new Student2
  var s: Student2 = null
  println(p.isInstanceOf[Student2])   //true
  //如果对象是 null,则 isInstanceOf 一定返回 false
  println(s.isInstanceOf[Student2])   //Flase
  if (p.isInstanceOf[Student2]){ // 判断 p 是否为 Student3 对象的实例
    s = p.asInstanceOf[Student2] //把 p 转换成 Student3 对象的实例
  }
  println(s.isInstanceOf[Student2])    //true

getClass和classOf

// isInstanceOf只能判断出对象是否是指定类以及其子类的对象,而不能精确判断出,对象就是指定类的对象
// 如果要求精确地判断对象就是指定类的对象,那么就只能使用getClass和classOf了
// 对象.getClass可以精确获取对象的类,classOf[类]可以精确获取类,然后使用==操作符即可判断
class Person2 {}
class Student2  extends Person2{}
测试:
  val p : Person2 = new Student2
  //判断p是否为Person2类的实例
  println(p.isInstanceOf[Person2])   //true
  //判断p的类型是否为Person2类
  println(p.getClass == classOf[Person2]) //Flase
  //判断p的类型是否为Student2类
  println(p.getClass == classOf[Student2])  //true

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

// 但是在实际开发中,比如spark的源码中,大量的地方都是使用了模式匹配的方式来进行类型的判断,这种方式更加地简洁明了,而且代码得可维护性和可扩展性也非常的高
// 使用模式匹配,功能性上来说,与isInstanceOf一样,也是判断主要是该类以及该类的子类的对象即可,不是精准判断的
class Person2 {}
class Student2  extends Person2{}
测试:
  val p : Person2 = new Student2
  p match{
    case per:Person2 => println("It's Person2's object")
    case _ =>println("unknown type")
  }
运行结果://It's Person2's object

protected

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

class Person2 {
  protected var name:String = "ynn"
  //protected[this] var hobby = "game"
  protected var hobby = "game"
}
class Student2  extends Person2{
  def sayHello = println("Hello, "+ name)
  def makeFriends(s : Student2): Unit ={
    println("my hobby is " + hobby + ", your hobby is " + s.hobby)
  }
}
测试: val p = new Student2
  	 p.makeFriends(p)
运行结果:加了protected[this]时,会报错:
Error:(6, 61) value hobby is not a member of com.doit.scc2.Student2
    println("my hobby is " + hobby + ", your hobby is " + s.hobby)
加了protected时,运行:my hobby is game, your hobby is game

调用父类的constructor

// Scala中,每个类可以有一个主constructor和任意多个辅助constructor,而每个辅助constructor的第一行都必须是调用其他辅助constructor或者是主constructor;
// 因此子类的辅助constructor是一定不可能直接调用父类的constructor的
// 只能在子类的主constructor中调用父类的constructor,以下这种语法,就是通过子类的主构造函数来调用父类的构造函数
// 注意!如果是父类中接收的参数,比如name和age,子类中接收时,就不要用任何val或var来修饰了,否则会认为是子类要覆盖父类的field
class Person4 (val name:String,val age:Int){}
class Student4 (name:String,age:Int,var score:Double) extends Person4(name,age){
  def this(name:String){
    this(name,0,0)
    println("this1 辅助constrctor")
  }
  def this(age:Int){
    this("ynn",age,0)
    println("this2 辅助constrctor")
  }
}
测试:
  val s1 = new Student4("ynn")    //this1 辅助constrctor
  val s2 = new Student4(20)    //this2 辅助constrctor
  val s3 = new Student4("ynn",20,2.0)   

匿名内部类

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

class Person5(protected val name:String) {
  def sayHello = "Hi" + name
}
object中写:
	val p = new Person5("ynn"){
    override def sayHello = "Hello , " + name
  }
  //定义一个方法,接受的方法参数就是Person类还有一个sayHello方法,和上边定义的匿名内部类结构一样
  def greeting(p:Person5{def sayHello:String}): Unit ={
    print(p.sayHello)
  }
  greeting(p)
运行结果:Hello , ynn

抽象类

// 如果在父类中,有某些方法无法立即实现,而需要依赖不同的子类来覆盖,重写实现自己不同的方法实现。
// 此时可以将父类中的这些方法不给出具体的实现,只有方法签名,这种方法就是抽象方法。
// 而一个类中如果有一个抽象方法(可以有具体的方法),那么类就必须用abstract来声明为抽象类,此时抽象类是不可以实例化的
//抽象类不一定有抽象方法,有抽象方法的类一定第抽象类
// 在子类中覆盖抽象类的抽象方法时,不需要使用override关键字
abstract class Person6(val name:String) {
  def sayHello:Unit
}
class Student5(name:String) extends Person6(name) {
  def sayHello:Unit = println("Hi,"+name)
}
测试:val s = new Student5("ynn")
  	s.sayHello
运行结果:Hi,ynn

// 如果在父类中,定义了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"
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值