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
类有两个成员变量 name
和 age
,以及一个方法 greet
。val
关键字表示这些变量是不可变的(即一旦初始化之后就不能再改变)。
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.
注意:
- 主构造器直接跟在类名后面,主构造器中的参数最后会被编译成字段/属性
- 主构造器执行时,会执行类中的所有语句
- 假设参数声明时不带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
- 第一次调用object的方法时,就会执行object的constructor,也就是object内部不在method中的代码;
- 但是object不能定义接受参数的constructor;
- 注意:object的constructor只会在其第一次被调用时执行一次,以后再次调用就不会再次执行constructor了;
- 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
实例的方法:apply
和 fromTuple
。
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进行声明,中文也可以称作样例类。
- case class其实有点类似于Java中的JavaBean的概念。即只定义field,并且由Scala编译时自动提供getter和setter方法,但是没有method.
- case class的主构造函数接收的参数通常不需要使用var或val修饰,
Scala自动就会使用val修饰 - 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 总结
- 类使用
class
关键字定义,可以包含成员变量、方法和构造函数。 - 对象使用
object
关键字定义,是单例的。 - 伴生对象与类同名,可以互相访问私有成员,通常用于提供工厂方法或静态方法。
通过这些概念,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
在这个例子中,num
和 str
被绑定到元组的元素上。
为什么有这种语法? 思考一下。因为只要使用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)!