自学内容网 自学内容网

3. Scala 高阶语法之面向对象和模式匹配

背景

上一节,介绍了scala中常见的数据结构:集合和元组。我们知道,scala既是面向对象,又是面向过程的,本文,我会介绍scala对于面向对象的支持以及scala中非常强大的功能模式匹配

1. 类与对象

在 Scala 中,类和对象是面向对象编程(OOP)的基本构建块。Scala 支持类似于 Java 的类定义,但还引入了一些额外的特性,使得类和对象的使用更加灵活和强大。以下是关于 Scala 中类和对象的一些关键概念和用法:

1.1 类

1.1.1 定义类

在 Scala 中,使用 class 关键字来定义一个类。以下是一个简单的类定义示例:

class Person(val name: String, val age: Int) {

  //定义属性
  var  name:String="wanlong";
  val  age:Int=18;
 
   def greet(): Unit = {
        println(s"Hello, my name is $name and I am $age years old.")
    }
}

在这个例子中,Person 类有两个成员变量 nameage,以及一个方法 greetval 关键字表示这些变量是不可变的(即一旦初始化之后就不能再改变)。

1.1.2 使用类创建对象

使用 new 关键字来创建类的实例:

val person = new Person("Alice", 30)
person.greet()  // 输出: Hello, my name is Alice and I am 30 years old.

1.1.3 主构造函数和辅助构造函数

Scala 中的主构造函数是在类定义中直接指定的参数列表。如果需要多个构造函数,可以通过定义 this 方法来实现:

class Person(val name: String, val age: Int) {
    def this(name: String) = this(name, 0)  // 辅助构造函数

    def greet(): Unit = {
        println(s"Hello, my name is $name and I am $age years old.")
    }
}

val child = new Person("Bob")  // 使用辅助构造函数
child.greet()  // 输出: Hello, my name is Bob and I am 0 years old.

注意:

  1. 主构造器直接跟在类名后面,主构造器中的参数最后会被编译成字段/属性
  2. 主构造器执行时,会执行类中的所有语句
  3. 假设参数声明时不带val或var,那么相当于private[this]只能在内部使用
package com.wanlong.next


class Person(val name:String,var age:Int){
  println("person constructor enter")

  val school="qinghua"

  println("person constructor leave")
}

object fTestConstructor extends App{
  //调用主构造器
  val person=new Person("wangwu",20)
  //在person外部使用,所以name和age 属性在构造器参数中 必须有val或者var修饰.否则编译报错
  
  println(person.name)
  println(person.age)
  println(person.school)

  //调整:修改主构造器的属性name修饰符为空,则报错,因为参数声明时不带va|或var,那么相当于private[this],只能在class内部使用:

}

运行结果:

person constructor enter
person constructor leave
wangwu
20
qinghua

1.2 对象(Object)

在 Scala 中,object 关键字用于定义单例对象。单例对象类似于 Java 中的静态类,但更加灵活和强大。 object相当于class的单个实例,通常在里面放一些静态的field或者method

  1. 第一次调用object的方法时,就会执行object的constructor,也就是object内部不在method中的代码;
  2. 但是object不能定义接受参数的constructor;
  3. 注意:object的constructor只会在其第一次被调用时执行一次,以后再次调用就不会再次执行constructor了;
  4. object通常用于作为单例模式的实现,或者放class的静态成员,比如工具方法。

1.2.1 定义单例对象

object Greeter {
    def greet(name: String): Unit = {
        println(s"Hello, $name!")
    }
}

Greeter.greet("Charlie")  // 输出: Hello, Charlie!

在这个例子中,Greeter 是一个单例对象,它有一个方法 greet。由于 Greeter 是单例的,因此在整个程序中只有一个 Greeter 实例。

1.2.2 伴生对象(Companion Object)

Scala 还允许定义一个类的伴生对象。伴生对象是与类同名的一个对象,伴生类和伴生对象,最大的特点就在于它们可以互相访问私有成员。这通常用于提供工厂方法或实现静态方法的功能。伴生类和伴生对象必须存放在一个.scala文件之中

class Point(val x: Int, val y: Int)

object Point {
    def apply(x: Int, y: Int): Point = new Point(x, y)
    def fromTuple(tuple: (Int, Int)): Point = new Point(tuple._1, tuple._2)
}

val p1 = Point(1, 2)  // 使用 apply 方法
val p2 = Point.fromTuple((3, 4))  // 使用 fromTuple 方法

在这个例子中,Point 类有一个伴生对象 Point,它提供了两种创建 Point 实例的方法:applyfromTuple

1.3 apply方法

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

在 Scala 中,apply 方法是一个具有特殊意义的方法,它允许对象像函数那样被调用。当你定义一个带有 apply 方法的对象或伴生对象时,你可以省略圆括号直接使用该对象或类名并传入参数,Scala 编译器会自动调用其 apply 方法。

1.3.1 定义 apply 方法

你可以在类、对象或特质(trait)中定义 apply 方法。以下是一些例子:

1.3.1.1 在对象中定义 apply 方法
object Factorial {
  def apply(n: Int): BigInt = {
    if (n <= 1) 1 else n * apply(n - 1)
  }
}

// 使用 Factorial 对象,像调用函数那样
val result = Factorial(5)  // 相当于 Factorial.apply(5)
println(result)  // 输出: 120

在这个例子中,Factorial 对象定义了一个 apply 方法,用于计算阶乘。当你使用 Factorial(5) 时,实际上是在调用 Factorial.apply(5)

1.3.1.2 在类的伴生对象中定义 apply 方法
class Point(val x: Int, val y: Int)

object Point {
  def apply(x: Int, y: Int): Point = new Point(x, y)
}

// 使用 Point 伴生对象的 apply 方法
val p = Point(1, 2)  // 相当于 Point.apply(1, 2)

在这个例子中,Point 类有一个伴生对象 Point,它定义了一个 apply 方法来创建 Point 实例。这使得创建 Point 实例更加简洁。

1.3.2 apply 方法的应用场景

  • 工厂方法apply 方法通常用作工厂方法,用于创建类的实例。它提供了一种更简洁、更函数式的方式来创建对象。
  • 使对象像函数那样被调用:通过定义 apply 方法,你可以使对象具有类似函数的行为,从而增加代码的灵活性和可读性。
  • 伴生对象中的 apply 方法:在伴生对象中定义 apply 方法是 Scala 中的一种常见模式,它允许你通过类名直接创建实例,而无需显式调用构造函数。

1.3.3 注意事项

  • apply 方法不是必须的,但它是 Scala 中实现工厂模式和使对象具有函数式行为的一种有用方式。
  • 当你在类名或对象名后直接跟参数列表时,Scala 编译器会查找并调用相应的 apply 方法(如果存在)。
  • apply 方法可以有多个参数列表,就像普通的 Scala 方法一样。

1.4 main方法

如同java中,如果要运行一个程序,必须编写一个包含main方法类一样;在scala中,如果要运行一个应用程序,必须有一个main方法,作为入口;
scala中的main方法定义为def main(args: Array[String]),而且必须定义在object中。
main方法的参数是一个数组,数组的元素是字符串;
scala中,main方法是程序的入口;
scala中,main方法没有返回值;
scala中,main方法的参数是一个数组,数组的元素是字符串;

package com.wanlong.next

object aTestMain {
  def main(args: Array[String]): Unit = {
    println("Hello World")
  }
}

1.5 caseClass 类

Scala中提供了一种特殊的类,用case class进行声明,中文也可以称作样例类。

  1. case class其实有点类似于Java中的JavaBean的概念。即只定义field,并且由Scala编译时自动提供getter和setter方法,但是没有method.
  2. case class的主构造函数接收的参数通常不需要使用var或val修饰,
    Scala自动就会使用val修饰
  3. Scala自动为case class定义了伴生对象,也就是object,并且定义了apply()方法,该方法接收主构造函数中相同的参数,并返回case class对象。
class Person
//构造器中每一个类型都为val 的参数,都会被编译成字段,不建议用var修饰
case class Teacher(name:String,subject:String) extends Person
case class Student(name:String,classroom:String) extends Person

//使用的时候
val wangwu = Teacher("wangwu", "math")
val zhangsan = Student("zhangsan", "class1")

1.6 总结

  1. 类使用 class 关键字定义,可以包含成员变量、方法和构造函数。
  2. 对象使用 object 关键字定义,是单例的。
  3. 伴生对象与类同名,可以互相访问私有成员,通常用于提供工厂方法或静态方法。

通过这些概念,Scala 提供了强大的面向对象编程支持,使得代码更加模块化和易于管理。

2. 模式匹配

模式匹配是Scala中非常有特色,非常强大的一种功能。模式匹配,其实类似于Java中的swichcase语法,即对一个值进行条件判断,然后针对不同的条件,进行不同的处理。但是Scala的模式匹配的功能比Java的swich case语法的功能要强大地多。

Java的swich case语去只能对值进行匹配。但是Scala的模式匹配除了可以对值进行匹配之外,还可以对类型进行匹配、如元组、数组、列表和自定义对象、对case class进行匹配、甚至对有值或没值(Option)进行匹配。

而且对于Spark来说,Scala的模式匹配功能也是极其重要的,在spark源码中大量地使用了模式工配功能。因此为了更好地编写Scala程序,并且更加通畅地看懂Spark的源码,学好模式匹配都是非常重要的。

Scala是没有Java中的switch case语法的,相对应的,Scala提供了更加强大的match case语法即模式匹配,match case的语法如下:变量 match{case 值 =>代码}。如果值为下划线,则代表了不满足以上所有情况下的默认情况如何处理。

此外,match case中,只要一个case分支满足并处理了就不会继续判断下一个case分支了。(与Java不同,java的switch case需要用break阻止),match case语法最基本的应用,就是对变量的值进行模式匹配

2.1 基本语法

val x = 1
x match {
  case 1 => println("One")
  case 2 => println("Two")
  case _ => println("Something else")
}

在这个例子中,x 的值会被匹配到不同的 case 分支,当匹配成功时,执行相应的代码块。

2.2 变量绑定

可以在模式匹配中绑定变量:

val pair = (2, "world")
pair match {
  case (1, str) => println(s"Hello, $str!")
  case (num, str) => println(s"$num $str")
}
//2 world

在这个例子中,numstr 被绑定到元组的元素上。
为什么有这种语法? 思考一下。因为只要使用case匹配到的值,是不是我们就知道这个值啦!!! 在这个case的处理语句中,是不是就直接可以使用写程序时就已知的值!

但是对于下划线_这种情况,所有不满足前面的case的值,都会进入_这种默认情况进行处理,此时如果我们在处理语句中需要拿到具体的值进行处理呢?那就需要使用这种在模式匹配中进行变量赋值的语法!!

2.3 类型模式

你可以根据类型进行匹配:

val x: Any = "Hello"
x match {
  case s: String => println(s"String: $s")
  case i: Int => println(s"Integer: $i")
  case _ => println("Unknown type")
}
//String: Hello

def judgeIdentify(p:Person) = p match {
  case Teacher(name,subject) => println("老师:"+name+","+subject)
  case Student(name,classroom) => println("学生:"+name+","+classroom)
  case _ => println("其他")
}

2.4 守卫条件

可以在 case 子句中添加守卫条件(guard conditions):

val num = 10
num match {
  case n if n < 0 => println("Negative")
  case n if n == 0 => println("Zero")
  case n if n > 0 => println("Positive")
}
//Positive

2.5 构造器模式

可以匹配自定义对象的构造器:

case class Person(name: String, age: Int)

val alice = Person("Alice", 30)
alice match {
  case Person(name, age) if age > 18 => println(s"$name is an adult.")
  case Person(name, age) => println(s"$name is a minor.")
}
//Alice is an adult.

2.6 序列模式

可以匹配序列(如列表和数组):

val list = List(1, 2, 3)
list match {
  case List(a, b, c) => println(s"First: $a, Second: $b, Third: $c")
  case _ => println("List does not have exactly three elements")
}
//First: 1, Second: 2, Third: 3

/**
* 模式匹配之Array & List
* 对Array进行模式匹配,分别可以匹配带有指定元素的数组、带有指定个数元素的数组、以某元素打头的数组
* 对List进行模式匹配,与Array类似,但是需要使用List特有的::操作符
*/
def greeting(arr:Array[String]): Unit ={
 arr match {
   //匹配带有指定元素的数组 这里匹配的是一个数组,数组中只有一个元素tom
   case Array("tom") => println("hello tom")
   //匹配带有指定个数元素的数组 这里匹配的是一个数组,数组中只有两个元素
   case Array(girl1,girl2) => println("hi "+girl1+" and "+girl2)
   //匹配以某元素打头的数组 这里匹配的是一个数组,数组中以hello开头 数组中至少有三个元素
   case Array("hello",_,_) => println("hello, nice to meet you")
   //匹配任意长度的数组 这里匹配的是一个数组,数组中以hello开头
   case Array("hello",_*) => println("hello, nice to meet you")
   case _ => println("unknow")
 }
}
    
def greeting2(list:List[String]): Unit = {
  list match {
    //匹配带有指定元素的List 这里匹配的是一个List,List中只有一个元素tom
    case  "tom" :: Nil => println("hello " + "tom")
    //匹配带有指定个数元素的List 这里匹配的是一个List,List中只有两个元素
    case girl1 :: girl2 :: Nil => println("hi " + girl1 + " and " + girl2)
    //匹配某个元素开头的List 这里匹配的是一个List,List中以leo开头
    case "leo" :: tail => println("hello,leo")
    case _ => println("unknow")
  }
}

2.7 元组模式

匹配元组:

val tuple = (1, "Scala")
tuple match {
  case (1, lang) => println(s"The language is $lang")
  case (num, lang) => println(s"Number: $num, Language: $lang")
}
//The language is Scala

def match_tuple(tuple:Any) = tuple match {
  //匹配第一个元素是1,第二个元素是2的Tuple
  case (1,2) => println("matched (1,2)")
  //匹配第一个元素是1,第二个元素是任意值的Tuple
  case(1,_) => println("matched (1,_)")
  //匹配第一个元素是任意值,第二个元素是2的Tuple
  case(_,2) => println("matched (_,2)")
  //匹配其他元素
  case _ => println("matched something else")
}

2.8 选项模式(Option)

处理 Option 类型:

val maybeNumber: Option[Int] = Some(42)
maybeNumber match {
  case Some(n) => println(s"Got a number: $n")
  case None => println("No number")
}
//Got a number: 42

2.9 密封特质(Sealed Traits)

使用密封特质进行穷举匹配,确保所有可能的情况都被覆盖:

sealed trait Shape
case class Circle(radius: Double) extends Shape
case class Rectangle(width: Double, height: Double) extends Shape
case object Square extends Shape

val shape: Shape = Rectangle(3.0, 4.0)
shape match {
  case Circle(r) => println(s"Circle with radius $r")
  case Rectangle(w, h) => println(s"Rectangle with width $w and height $h")
  case Square => println("Square")
}
//Rectangle with width 3.0 and height 4.0

以上,如有错误,请不吝指正。


原文地址:https://blog.csdn.net/wlyang666/article/details/145018652

免责声明:本站文章内容转载自网络资源,如侵犯了原著者的合法权益,可联系本站删除。更多内容请关注自学内容网(zxcms.com)!