Scala基础(二)——面向对象(中)

本文主要探讨Scala中的面向对象编程,包括对象的概念,如object、伴生对象和继承。详细阐述了object的特性,如单例模式实现、apply方法、main方法及用作枚举功能。同时介绍了继承的相关概念,如extends、override、super、isInstanceOf和asInstanceOf等,以及如何通过模式匹配进行类型判断和使用protected关键字。文章还提到了抽象类和抽象field的定义与覆盖规则。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、面向对象编程

1、对象

1)object

  • 相当于class的单个实例,通常在里面放一些静态的field或者method
  • 第一次调用object的方法时,就会执行object的constructor,也就是object内部不在method中的代码;但是object不能定义接受参数的constructor
  • 注意: object的constructor只会在其第一次被调用时执行一次,以后再次调用就不会再次执行constructor了
  • object通常用于作为单例模式的实现,或者放class的静态成员,比如工具方法
package com.doit.scala.class11

object Person_11_1 {
  private var eyeNum = 2
  print("this Person object !")
  def getEyeNum = eyeNum
}

//测试类:
package com.doit.scala.class11

object TestClass11 {
  def main(args: Array[String]) = {
    println(Person_11_1.getEyeNum)
  }
}

在这里插入图片描述

2)伴生对象

  • 如果有一个class,还有一个与class同名的object,那么就称这个object是class的伴生对象,class是object的伴生类
  • 伴生类和伴生对象必须存放在一个.scala文件之中
  • 伴生类和伴生对象,最大的特点就在于,互相可以访问private field
object Person_11_2 {
  private val eyeNum = 2
  def getEyeNum = eyeNum
}

class Person_11_2(val name: String, val age: Int){
  def sayHello = println("Hi," + name + ",I guess you are " + age + " years old !" + ",and usually you must have " + Person_11_2.eyeNum + " eyes.")
}

//测试类
object TestClass11 {
  def main(args: Array[String]) = {
    val p = new Person_11_2("wang", 23)
    p.sayHello
  }
}

在这里插入图片描述

3)让object继承抽象类

  • object的功能其实和class类似,除了不能定义接受参数的constructor之外
  • object也可以继承抽象类,并覆盖抽象类中的方法
abstract class Hello_11_3 (var message: String){
  def sayHello(name: String)
}

object Hellolmpl_11_3 extends Hello_11_3 ("hello"){
  override def sayHello(name: String): Unit = {
    println(message + "," + name)
  }
}

//测试类
object TestClass11 {
  def main(args: Array[String]) = {
    Hellolmpl_11_3.sayHello("xiaoming")
  }
}

4)apply方法

  • object中非常重要的一个特殊方法,就是apply方法
  • 通常在伴生对象中实现apply方法,并在其中实现构造伴生类的对象的功能
  • 而创建伴生类的对象时,通常不会使用new Class的方式,而是使用Class()的方式,隐式地调用伴生对象得apply方法,这样会让对象创建更加简洁
  • 比如,Array类的伴生对象的apply方法就实现了接收可变数量的参数,并创建一个Array对象的功能
    val a = Array(1, 2, 3, 4, 5)
//定义自己的伴生类和伴生对象
class Person_11_4 (val name: String){
  println("你就new把")
}

object Person_11_4{
  def apply(name: String) = {
    println("别new了!!!")
    new Person_11_4(name)
  }
}

//测试类
object TestClass11 {
  def main(args: Array[String]) = {
    val p = new Person_11_4("wang")
    val p1 = Person_11_4("zhang")
  }
}

在这里插入图片描述

5)main方法

  • 就如同java中,如果要运行一个程序,必须编写一个包含main方法类一样;在scala中,如果要运行一个应用程序,那么必须有一个main方法,作为入口
  • scala中的main方法定义为def main(args: Array[String]),而且必须定义在object中

在E盘创建一个HelloWorld.scala,内容如下,在windows上进入cmd,执行scalac HelloWorld.scala编译scala文件 ,产生一个HelloWorld.class,然后scala HelloWorld

object HelloWorld {
  def main(args: Array[String]) {
    println("Hello World!!!")
  }
}

在这里插入图片描述
//自动生成.class文件
在这里插入图片描述
在这里插入图片描述
除了自己实现main方法之外,还可以继承App Trait,然后将需要在main方法中运行的代码,直接作为object的constructor代码;而且用args可以接受传入的参数

object HelloWord_11_5 extends App{
  if(args.length > 0) println("hello," + args(0))
  else  println("Hello World!!!")
}

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
//再次运行,得到hello,1

如果要运行上述代码,需要将其放入.scala文件,然后先使用scalac编译,再用scala执行

scalac HelloWorld.scala
scala -Dscala.time HelloWorld

App Trait的工作原理: App Trait继承自DelayedInit Trait,scalac命令进行编译时,会把继承App Trait的object的constructor代码都放到DelayedInit Trait的delayedInit方法中执行

6) 用object来实现枚举功能

  • Scala没有直接提供类似于Java中的Enum这样的枚举特性,如果要实现枚举,则需要用object继承Enumeration类,并且调用Value方法来初始化枚举值
object Season_11_7_1 extends Enumeration {
  val SPRING, SUMMER, AUTUMN, WINTER = Value
}

//测试类
object TestClass11 {
  def main(args: Array[String]) = {
    println(Season_11_7_1.withName("SPRING"))
  }
}

//还可以通过Value传入枚举值的id和name,通过id和toString可以获取; 还可以通过id和name来查找枚举值
object Season_11_7_2 extends Enumeration {
  val SPRING = Value(0, "spring")
  val SUMMER = Value(1, "summer")
  val AUTUMN = Value(2, "autumn")
  val WINTER = Value(3, "winter")
}

//测试类
object TestClass11 {
  def main(args: Array[String]) = {
    println(Season_11_7_1.withName("SPRING"))
    println(Season_11_7_2(0))
    //使用枚举object.values可以遍历枚举值
    for(ele <- Season_11_7_2.values) print (ele + " ")
  }
}

在这里插入图片描述

2、继承

1)extends

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

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

//子类
package com.doit.scala.class12

class Student_12_1 extends Person_12_1 {
  private var score = "A"
  def getScore = score
}

//测试类
package com.doit.scala.class12

object TestClass12 {
  def main(args: Array[String]): Unit = {
    val s = new Student_12_1
    println(s.getName)
    println(s.getScore)
  }
}

在这里插入图片描述

2)override和super

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

//父类
class Person_12_2 {
  private var name = "leo"
  def getName = name
}

//子类
class Student_12_2 extends Person_12_2{
  private var score = "A"
  def getScore = score
  override def getName = "Hi,I'm student my name is " + super.getName
}

//测试类
package com.doit.scala.class12

object TestClass12 {
  def main(args: Array[String]): Unit = {
    val s = new Student_12_2
    println(s.getName)
    println(s.getScore)
  }
}

在这里插入图片描述

3)override field

  • Scala中,子类可以覆盖父类的val field,而且子类的val field还可以覆盖父类的val field的getter方法;只要在子类中使用override关键字即可
class Person_12_3 {
  val name: String = "Person"
  def age: Int = 0
}

class Student_12_3 extends Person_12_3 {
  override val name: String = "leo"
  override def age: Int = 30
}

object TestClass12 {
  def main(args: Array[String]): Unit = {
    val s = new Student_12_3
    println(s.name)
    println(s.age)
  }
}

在这里插入图片描述

4) isInstanceOf和asInstanceOf

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

class Student_12_4 extends Person_12_4 {
  val nameA = "wang"
}

//测试类
object TestClass12 {
  def main(args: Array[String]): Unit = {
    val p: Person_12_4 = new Student_12_4
    print(p.name)
    var s: Student_12_4 = null
    println(p.isInstanceOf[Student_12_4])
    s = p.asInstanceOf[Student_12_4]
    println(s.nameA)
  }
}

在这里插入图片描述

5)getClass和classOf

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

class Student_12_5 extends Person_12_5{
  
}

//测试类
object TestClass12 {
  def main(args: Array[String]): Unit = {
    val p: Person_12_5 = new Student_12_5
    p.isInstanceOf[Person_12_5]
    println(p.getClass == classOf[Person_12_5])
    println(p.getClass == classOf[Student_12_5])
  }
}

在这里插入图片描述

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

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

class Student_12_6 {}

object TestClass12 {
  def main(args: Array[String]): Unit = {
    val p = new Person_12_6
    p match {
      case per: Person_12_6 => println("is a person_12_6")
      case per1: Person_12_5 => println("is a person_12_5")
      case _ => println("unknown type")
    }
  }
}

在这里插入图片描述

7)protected

  • 跟java一样,scala中同样可以使用protected关键字来修饰field和method,这样在子类中就不需要super关键字,直接就可以访问field和method
  • 还可以使用protected[this],则只能在当前子类对象中访问父类的field和method,无法通过其他子类对象访问父类的field和method
class Person_12_7 {
  protected var name: String = "leo"
  protected[this] var hobby: String = "game"
}

class Student_12_7 extends Person_12_7{
  def sayHello = println("Hello," + name)
  def makeFriends(s: Student_12_7): Unit ={
    println("my hobby is %s, your hobby is %s".format(hobby, s.hobby))
  }
}

8)调用父类的constructor

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

class Student_12_8(name: String,age: Int, var score: Double) extends Person_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 TestClass12 {
  def main(args: Array[String]): Unit = {
    val s1 = new Student_12_8("wang", 30, 2.3)
    val s2 = new Student_12_8("liu")
    val s3 = new Student_12_8(24)
    val s4 = new Student_12_8(33,4.5)
  }
}

9)匿名内部类

  • 在Scala中,匿名子类是非常常见,而且非常强大的。
  • 匿名子类,也就是说,可以定义一个类的没有名称的子类,并直接创建其对象,然后将对象的引用赋予一个变量。之后甚至可以将该匿名子类的对象传递给其他函数。
class Person_12_9 (protected var name: String){
  def sayHello = "Hello,I'm " + name
}

object TestClass12 {
  def main(args: Array[String]): Unit = {
    val p1 = new Person_12_9("leo"){
      override def sayHello: String = "Hi " + name
    }
    val p2 = new Person_12_9("leo"){
      override def sayHello: String = "h " + name
    }
    def greeting(p: Person_12_9{def sayHello: String}): Unit ={
      println(p.sayHello)
    }
    greeting(p1)
    greeting(p2)
  }
}

在这里插入图片描述

10)抽象类

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


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


object TestClass12 {
  def main(args: Array[String]): Unit = {
    val s : Person_12_10 = new Student_12_10("wang")
    val s1 = new Student_12_10("wang")
    s1.sayHello
    s.sayHello
    s.sayHi()
    s1.sayHi()
  }
}

在这里插入图片描述

11)抽象field

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值