scala函数高级

1scala 函数高级操作

1、scala 高阶函数
2 、匿名函数
3、柯里化、闭包
4、控制抽象
5、自定义While关键字
6、惰性加载

2.1 高阶函数

在 Scala 中,函数是一等公民。怎么体现的呢?
对于一个函数我们可以:定义函数、调用函数

object TestFunction {
 def main(args: Array[String]): Unit = {
 // 调用函数
foo()
 }
 // 定义函数
 def foo():Unit = {
 println("foo...")
 } }

但是其实函数还有更高阶的用法

1)函数可以作为值进行传递

object TestFunction {
 def main(args: Array[String]): Unit = {
 //(1)调用 foo 函数,把返回值给变量 f
 //val f = foo()
 val f = foo
 println(f)
 //(2)在被调用函数 foo 后面加上 _,相当于把函数 foo 当成一个整体,
传递给变量 f1
 val f1 = foo _
 foo()
 f1()


//(3)如果明确变量类型,那么不使用下划线也可以将函数作为整体传递给
变量
var f2:()=>Int = foo 
 }
 def foo():Int = {
 println("foo...")
 1
 } }

2)函数可以作为参数进行传递

def main(args: Array[String]): Unit = {
 
 // (1)定义一个函数,函数参数还是一个函数签名;f 表示函数名称;(Int,Int)
表示输入两个 Int 参数;Int 表示函数返回值
 def f1(f: (Int, Int) => Int): Int = {
 f(2, 4)
 }
 
 // (2)定义一个函数,参数和返回值类型和 f1 的输入参数一致
 def add(a: Int, b: Int): Int = a + b
 
 // (3)将 add 函数作为参数传递给 f1 函数,如果能够推断出来不是调用,_
可以省略
 println(f1(add))
println(f1(add _))
//可以传递匿名函数
}

3)函数可以作为函数返回值返回

def main(args: Array[String]): Unit = {
 def f1() = {
 def f2() = {
}
f2 _
}
val f = f1()
// 因为 f1 函数的返回值依然为函数,所以可以变量 f 可以作为函数继续调用
f()
// 上面的代码可以简化为
f1()()}

2.2 匿名函数

1)说明

没有名字的函数就是匿名函数。
(x:Int)=>{函数体} x:表示输入参数类型;Int:表示输入参数类型;函数体:表示具体代码逻辑

2)案例实操

需求 1:传递的函数有一个参数
传递匿名函数至简原则:
(1)参数的类型可以省略,会根据形参进行自动的推导
(2)类型省略之后,发现只有一个参数,则圆括号可以省略;其他情况:没有参数和参
数超过 1 的永远不能省略圆括号。 (3)匿名函数如果只有一行,则大括号也可以省略
(4)如果参数只出现一次,则参数省略且后面参数可以用_代替

 def main(args: Array[String]): Unit = {
 // (1)定义一个函数:参数包含数据和逻辑函数
 def operation(arr: Array[Int], op: Int => Int) = {
 for (elem <- arr) yield op(elem)
 }
 // (2)定义逻辑函数
 def op(ele: Int): Int = {
 ele + 1
 }
 // (3)标准函数调用
 val arr = operation(Array(1, 2, 3, 4), op)
 println(arr.mkString(","))
 // (4)采用匿名函数
 val arr1 = operation(Array(1, 2, 3, 4), (ele: Int) => {
 ele + 1
 })
 println(arr1.mkString(","))
 // (4.1)参数的类型可以省略,会根据形参进行自动的推导;
 val arr2 = operation(Array(1, 2, 3, 4), (ele) => {
 ele + 1
 })
 println(arr2.mkString(","))
 // (4.2)类型省略之后,发现只有一个参数,则圆括号可以省略;其他情
况:没有参数和参数超过 1 的永远不能省略圆括号。
 val arr3 = operation(Array(1, 2, 3, 4), ele => {
 ele + 1
 })
 println(arr3.mkString(","))
 // (4.3) 匿名函数如果只有一行,则大括号也可以省略
 val arr4 = operation(Array(1, 2, 3, 4), ele => ele + 1)
 println(arr4.mkString(","))
 //(4.4)如果参数只出现一次,则参数省略且后面参数可以用_代替
 val arr5 = operation(Array(1, 2, 3, 4), _ + 1)
 println(arr5.mkString(","))
 } }

需求 2:传递的函数有两个参数

object TestFunction {
 def main(args: Array[String]): Unit = {
 def calculator(a: Int, b: Int, op: (Int, Int) => Int): Int 
= {
 op(a, b)
 }
 // (1)标准版
 println(calculator(2, 3, (x: Int, y: Int) => {x + y}))
 // (2)如果只有一行,则大括号也可以省略
 println(calculator(2, 3, (x: Int, y: Int) => x + y))
 // (3)参数的类型可以省略,会根据形参进行自动的推导;
 println(calculator(2, 3, (x , y) => x + y))
 // (4)如果参数只出现一次,则参数省略且后面参数可以用_代替
 println(calculator(2, 3, _ + _))
 } }

2.3 高阶函数案例

需求:模拟 Map 映射、Filter 过滤、Reduce 聚合

object TestFunction {
 def main(args: Array[String]): Unit = {
 // (1)map 映射
 def map(arr: Array[Int], op: Int => Int) = {
 for (elem <- arr) yield op(elem)
 }
 val arr = map(Array(1, 2, 3, 4), (x: Int) => {
 x * x
 })
 println(arr.mkString(","))
 // (2)filter 过滤。有参数,且参数再后面只使用一次,则参数省略且
后面参数用_表示
 def filter(arr:Array[Int],op:Int =>Boolean) ={
 var arr1:ArrayBuffer[Int] = ArrayBuffer[Int]()
 for(elem <- arr if op(elem)){
 arr1.append(elem)
 }
 arr1.toArray
 }
 var arr1 = filter(Array(1, 2, 3, 4), _ % 2 == 1)
 println(arr1.mkString(","))
 // (3)reduce 聚合。有多个参数,且每个参数再后面只使用一次,则参
数省略且后面参数用_表示,第 n 个_代表第 n 个参数
 def reduce(arr: Array[Int], op: (Int, Int) => Int) = {
 var init: Int = arr(0)
 for (elem <- 1 until arr.length) {
 init = op(init, elem)
 }
 init
 }
 //val arr2 = reduce(Array(1, 2, 3, 4), (x, y) => x * y)
 val arr2 = reduce(Array(1, 2, 3, 4), _ * _)
 println(arr2)
 } }

2.4 函数柯里化&闭包

闭包:函数式编程的标配

1)说明

闭包:如果一个函数,访问到了它的外部(局部)变量的值,那么这个函数和他所处的
环境,称为闭包
函数柯里化:把一个参数列表的多个参数,变成多个参数列表。 2)案例实操
(1)闭包
object TestFunction {
def main(args: Array[String]): Unit = {
def f1()={
var a:Int = 10
def f2(b:Int)={
a + b
}
f2 _
}
// 在调用时,f1 函数执行完毕后,局部变量 a 应该随着栈空间释放掉
val f = f1()
// 但是在此处,变量 a 其实并没有释放,而是包含在了 f2 函数的内部,形
成了闭合的效果
println(f(3))

println(f1()(3))
// 函数柯里化,其实就是将复杂的参数逻辑变得简单化,函数柯里化一定存
在闭包
def f3()(b:Int)={
a + b
}
println(f3()(3))
} }

2.5 控制抽象

1)值调用:把计算后的值传递过去

object TestControl {
 def main(args: Array[String]): Unit = {
 def f = ()=>{
 println("f...")
  10
 }
 foo(f())
 }
 def foo(a: Int):Unit = {
 println(a)
 println(a)
 } }

2)名调用:把代码传递过去

object TestControl {
 def main(args: Array[String]): Unit = {
 def f = ()=>{
 println("f...")
 10
 }
 foo(f())
 }
//def foo(a: Int):Unit = {
 def foo(a: =>Int):Unit = {//注意这里变量 a 没有小括号了
 println(a)
 println(a)
 } }

输出结果:
f…
10
f…
10
注意:Java 只有值调用;Scala 既有值调用,又有名调用。

3)案例实操

object TestFunction {
 def main(args: Array[String]): Unit = {
 // (1)传递代码块
 foo({
 println("aaa")
 })
 // (2)小括号可以省略
 foo{
 println("aaa")
 }
 }
 def foo(a: =>Unit):Unit = {
 println(a)
 println(a)
 } }

2.6 自定义while

//方法1

def myWhile(condition : =>Boolean): (=>Unit) => Unit ={

    def loop(op : =>Unit): Unit ={

      if (condition){
        op
        myWhile(condition)(op)
      }

    }
    loop
  }
  // 方法2
  def myWhile2(condition : =>Boolean): (=>Unit) => Unit ={

    op  =>{
      if (condition){
        op
        myWhile(condition)(op)
      }
    }

  }

//方法3 使用柯里化&闭包
  def myWhile3(condition : =>Boolean)(op : =>Unit) ={
    if (condition){
      op
      myWhile(condition)(op)
    }
  }

2.7 惰性加载

1)说明
当函数返回值被声明为 lazy 时,函数的执行将被推迟,直到我们首次对此取值,该函
数才会执行。这种函数我们称之为惰性函数。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值