Scala 中下划线的用途

本文详细介绍了Scala中下划线的各种应用场景,包括存在性类型、高阶类型参数、临时变量、临时参数等,并通过具体例子展示了下划线如何在函数式编程中提高代码的简洁性和可读性。

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

转载自: https://my.oschina.net/leejun2005/blog/405305

Scala 作为一门函数式编程语言,对习惯了指令式编程语言的同学来说,会不大习惯,这里除了思维方式之外,还有语法层面的,比如 underscore(下划线)就会出现在多种场合,令初学者相当疑惑,今天就来总结下 Scala 中下划线的用法。

1、存在性类型:Existential types
def foo(l: List[Option[_]]) = ...

2、高阶类型参数:Higher kinded type parameters
case class A[K[_],T](a: K[T])

3、临时变量:Ignored variables
val _ = 5

4、临时参数:Ignored parameters
List(1, 2, 3) foreach { _ => println("Hi") }

5、通配模式:Wildcard patterns
Some(5) match { case Some(_) => println("Yes") }
match {
case List(1,_,_) => " a list with three element and the first element is 1"
case List(_*)  => " a list with zero or more elements "
case Map[_,_] => " matches a map with any key type and any value type "
case _ =>
}
val (a, _) = (1, 2)
for (_ <- 1 to 10)

6、通配导入:Wildcard imports
import java.util._

7、隐藏导入:Hiding imports
// Imports all the members of the object Fun but renames Foo to Bar
import com.test.Fun.{ Foo => Bar , _ }

// Imports all the members except Foo. To exclude a member rename it to _
import com.test.Fun.{ Foo => _ , _ }

8、连接字母和标点符号:Joining letters to punctuation
def bang_!(x: Int) = 5

9、占位符语法:Placeholder syntax
List(1, 2, 3) map (_ + 2)
_ + _    ( (_: Int) + (_: Int) )(2,3)

val nums = List(1,2,3,4,5,6,7,8,9,10)

nums map (_ + 2)
nums sortWith(_>_)
nums filter (_ % 2 == 0)
nums reduceLeft(_+_)
nums reduce (_ + _)
nums reduceLeft(_ max _)
nums.exists(_ > 5)
nums.takeWhile(_ < 8)

10、偏应用函数:Partially applied functions
def fun = {     // Some code
}
val funLike = fun _

List(1, 2, 3) foreach println _

1 to 5 map (10 * _)

//List("foo", "bar", "baz").map(_.toUpperCase())
List("foo", "bar", "baz").map(n => n.toUpperCase())

11、初始化默认值:default value
var i: Int = _

12、作为参数名:
//访问map
var m3 = Map((1,100), (2,200))
for(e<-m3) println(e._1 + ": " + e._2)
m3 filter (e=>e._1>1)
m3 filterKeys (_>1)
m3.map(e=>(e._1*10, e._2))
m3 map (e=>e._2)

//访问元组:tuple getters
(1,2)._2

13、参数序列:parameters Sequence  _*作为一个整体,告诉编译器你希望将某个参数当作参数序列处理。例如val s = sum(1 to 5:_*)就是将1 to 5当作参数序列处理。
//Range转换为List
List(1 to 5:_*)

//Range转换为Vector
Vector(1 to 5: _*)

//可变参数中
def capitalizeAll(args: String*) = {   args.map { arg =>     arg.capitalize   }
}

val arr = Array("what's", "up", "doc?")
capitalizeAll(arr: _*)

这里需要注意的是,以下两种写法实现的是完全不一样的功能:

foo _               // Eta expansion of method into method value

foo(_)              // Partial function application

Example showing why foo(_) and foo _ are different:

trait PlaceholderExample {   def process[A](f: A => Unit)   val set: Set[_ => Unit]   set.foreach(process _) // Error    set.foreach(process(_)) // No Error
}

In the first case, process _ represents a method; Scala takes the polymorphic method and attempts to make it monomorphic by filling in the type parameter, but realizes that there is no type that can be filled in for A that will give the type (_ => Unit) => ? (Existential _ is not a type).

In the second case, process(_) is a lambda; when writing a lambda with no explicit argument type, Scala infers the type from the argument that foreach expects, and _ => Unit is a type (whereas just plain _ isn't), so it can be substituted and inferred.

This may well be the trickiest gotcha in Scala I have ever encountered.

Refer:

[1] What are all the uses of an underscore in Scala?

http://stackoverflow.com/questions/8000903/what-are-all-the-uses-of-an-underscore-in-scala

[2] Scala punctuation (AKA symbols and operators)

http://stackoverflow.com/questions/7888944/scala-punctuation-aka-symbols-and-operators/7890032#7890032

[3] Scala中的下划线到底有多少种应用场景?

http://www.zhihu.com/question/21622725

[4] Strange type mismatch when using member access instead of extractor

http://stackoverflow.com/questions/9610736/strange-type-mismatch-when-using-member-access-instead-of-extractor/9610961

[5] Scala简明教程

http://colobu.com/2015/01/14/Scala-Quick-Start-for-Java-Programmers/

<think>好的,用户想了解Scala中函数和方法的区别以及如何定义。首先,我需要回忆Scala的基础知识,确保信息准确。记得函数和方法在Scala中是不同的概念,但有时候容易混淆。 首先,方法的定义是使用def关键字,属于类、特质或对象的一部分,通常与面向对象编程相关。而函数是一个独立的代码块,可以赋值给变量,作为参数传递或作为返回值,是一等公民。这点需要明确区分。 接下来,应该从定义方式入手。比如,方法需要在类或对象中定义,而函数可以通过lambda表达式或FunctionN的实例来创建。例如,方法可能有参数列表和返回类型,而函数可以作为变量存储。 然后,作用域也是一个区别点。方法在类或对象内部定义,而函数可以在任何地方定义,包括另一个函数内部。比如,用户提供的引用[2]中的例子,函数sayHello定义在main方法内部,这说明函数可以在方法内定义。 调用方式的不同也需要说明。方法调用依赖于对象实例,而函数可以直接调用或传递。此外,方法可以转换成函数,比如通过方法名加空格和下划线的方式,如method _,这样可以将方法提升为函数。 关于定义示例,用户给的引用[2]中的代码展示了在对象内部定义方法和函数的情况。比如,main方法中定义了函数sayHello,而方法可能定义在类或对象的结构体中。需要分别给出方法和函数的定义例子,比如在对象中定义一个方法,以及在方法内部或外部定义一个函数。 可能还需要提到,函数可以作为参数传递给高阶函数,如map、filter等,而方法需要先转换成函数才能使用。这点对于理解函数式编程在Scala中的应用很重要。 用户的问题还可能涉及它们的使用场景,比如在函数式编程中更常用函数,而在面向对象中更常用方法。但Scala是混合范式,所以两者都可以灵活使用。 需要检查是否有遗漏的点,比如类型推断、参数列表的差异。例如,方法可以有多参数列表,而函数的参数必须在一个列表中。此外,方法可以有隐式参数,而函数可能不支持。 最后,确保回答结构清晰,分点说明区别,并提供定义示例。同时,根据用户提供的引用内容,正确引用相关部分,如引用[1]提到函数是一等公民,可以在任何地方定义,作为参数和返回值,这点需要强调。而引用[2]中的代码示例展示了函数定义和调用的具体方式,应该结合这些例子进行说明。 总结回答的结构:先解释方法和函数的定义,然后分点列出区别,包括定义位置、语法、作用域、转换方式等,再给出各自的代码示例,最后总结两者的主要区别和应用场景。同时,生成相关问题,帮助用户深入理解。</think>### Scala函数与方法的区别及定义方式 #### 一、定义方式 1. **方法** 使用`def`关键字定义,属于类、特质或对象的成员,具有面向对象特性。 **示例**: ```scala class Calculator { def add(a: Int, b: Int): Int = a + b // 类中定义方法 } ``` 2. **函数** 通过`val`或`var`定义,是独立的值(类型为`FunctionN`),可赋值给变量或作为参数传递[^1]。 **示例**: ```scala val multiply = (a: Int, b: Int) => a * b // 函数赋值给变量 ``` #### 二、核心区别 | 特性 | 方法 | 函数 | |-------------|-------------------------------|-------------------------------| | **定义位置** | 类/对象内部 | 任何地方(包括函数内部)[^2] | | **语法** | `def`关键字 | `=>`箭头表达式 | | **独立性** | 依附于对象 | 一等公民,可独立存在[^1] | | **参数列表** | 支持多参数列表(柯里化) | 单参数列表 | | **转换方式** | 可通过`方法名 _`转为函数 | 无需转换直接使用 | #### 三、代码示例对比 1. **方法定义与调用** ```scala object MathUtils { def square(x: Int): Int = x * x // 对象中的方法 } MathUtils.square(5) // 输出25 ``` 2. **函数定义与传递** ```scala val greet: String => Unit = (name: String) => println(s"Hello, $name") List("Alice", "Bob").foreach(greet) // 函数作为参数传递 ``` 3. **方法转函数** ```scala def methodToFunction(x: Int): Int = x * 2 val func = methodToFunction _ // 方法转为函数 func(3) // 输出6 ``` #### 四、核心总结 - **方法**是面向对象的基本单元,**函数**是函数式编程的核心。 - 函数可灵活用于高阶操作(如`map`、`filter`),方法需转换后才能使用。 - 方法支持更复杂的语法(如柯里化),函数更轻量级。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值