函数式编程基础

函数定义和高阶函数

函数字面量
  • 字面量包括整数字面量、浮点数字面量、布尔型字面量、字符字面 量、字符串字面量、符号字面量、函数字面量和元组字面量。

scala> val i = 123
i: Int = 123

scala> val i = 3.14
i: Double = 3.14

scala> val i = true
i: Boolean = true

scala> val i = 'A'
i: Char = A

scala> val i = "Hello"
i: String = Hello
  • 除了函数字面量我们会比较陌生以外,其他几种字面量都很容易理解。

  • 函数字面量可以体现函数式编程的核心理念

  • 在非函数式编程语言里,函数的定义包含了“函数类型”和“值”两种层 面的内容

  • 但是,在函数式编程中,函数是“头等公民”,可以像任何其他数据类型 一样被传递和操作,也就是说,函数的使用方式和其他数据类型的使用方 式完全一致

  • 这时,我们就可以像定义变量那样去定义一个函数,由此导致的结果是, 函数也会和其他变量一样,开始有“值”

  • 就像变量的“类型”和“值”是分开的两个概念一样,函数式编程中,函数的“类型”和“值”也成为两个分开的概念,函数的“值”,就是“函数字面量"

  • 下面我们一点点引导大家更好地理解函数的“类型”和“值”的概念。 我们现在定义一个大家比较熟悉的传统类型的函数,定义的语法和我们之前介绍 过的定义“类中的方法”类似(实际上,定义函数最常用的方法是作为某个对象的成员,这种函数被称为方法):

函数的类型和值
匿名函数、Lamda表达式与闭包
  • 我们不需要给每个函数命名,这时就可以使用匿名函数,如下:

scala> (num: Int) => num*2
res1: Int => Int = <function1>
  • 上面这种匿名函数的定义形式,我们经常称为“Lamda表达式”。“Lamda 表达式”的形式如下:

(参数) => 表达式 //如果参数只有一个,参数的圆括号可以省略
  • 我们可以直接把匿名函数存放到变量中,下面是在Scala解释器中的执行过程:

scala> val myNumFunc: Int => Int = (num: Int) => num*2 //这行是我们输入的命令,把匿名函数定义成一个值,赋值给myNumFunc变量
myNumFunc: Int => Int = <function1> //这行是执行结果
scala> println(myNumFunc(3)) //myNumFunc函数调用的时候,需要给出参数的值,这里传入3,得到乘法结果是6
6
  • 实际上,Scala具有类型推断机制,可以自动推断变量类型,比如下面两条语句都是可以的:

scala> val number: Int = 7
number: Int = 7
scala> val number = 7
number: Int = 7
  • 所以,上面的定义中,我们可以myNumFunc的类型声明,也就是去掉 “Int=>Int”,在Scala解释器中的执行过程如下:

scala> val myNumFunc = (num: Int) => num*2
myNumFunc: Int => Int = <function1>
scala>  println(myNumFunc(36))
72
  • 下面我们尝试一下,省略num的类型声明,但是,给出myNumFunc的类型 声明,在Scala解释器中的执行过程如下:

scala> val myNumFunc: Int => Int = (num) => num*2
myNumFunc: Int => Int = <function1>
  • 不会报错,因为,给出了myNumFunc的类型为“Int=>Int”以后,解释器可 以推断出num类型为Int类型。

  • 闭包是一个函数,一种比较特殊的函数,它和普通的函数有很大区别

  • 普通函数:

scala> var more = 1
more: Int = 1
scala> val addMore = (x: Int) => x > 0
addMore: Int => Boolean = <function1>
scala> addMore(10)
res11: Boolean = true
  • 闭包:

scala> var more = 1
more: Int = 1
scala> val addMore = (x: Int) => x + more
addMore: Int => Int = <function1>
scala> addMore(10)
res5: Int = 11
scala> more = 9
more: Int = 9
scala> addMore(10)
res10: Int = 19
  • 每次addMore函数被调用时都会创建 一个新闭包

  • 每个闭包都会访问闭包创建时活跃的 more变量

占位符语法
  • 为了让函数字面量更加简洁,我们可以使用下划线作为一个或多个参数的占 位符,只要每个参数在函数字面量内仅出现一次。

scala> val numList = List(-3,-5,1,6,9)
numList: List[Int] = List(-3, -5, 1, 6, 9)
scala> numList.filter(x => x > 0)
res0: List[Int] = List(1, 6, 9)
scala>  numList.filter(_ > 0)
res1: List[Int] = List(1, 6, 9)
  • 从上面运行结果可以看出,下面两个函数字面量是等价的。

x => x > 0
_ > 0
  • 有时你把下划线当作参数的占位符时,编译器有可能没有足够的信息推断缺 失的参数类型。例如,假设你只是写_ + _:

scala> val f = _ + _
<console>:7: error: missing parameter type for expanded function ((x$1, x$2) => x$1.$plus(x$2))
      val f = _ + _
              ^
<console>:7: error: missing parameter type for expanded function ((x$1: <error>, x$2) => x$1.$plus(x$2))
      val f = _ + _
                  ^
  • 这种情况下,你可以运用 冒号指定类型,如下:

scala> val f = (_: Int) + (_: Int)
f: (Int, Int) => Int = <function2>
scala> f(5,8)
res2: Int = 13
  • 请留心 _ + _将扩展成带两个参数的函数字面量。这也是仅当每个参数在函数 字面量中最多出现一次的情况下你才能运用 这种短格式的原由 。多个下划线 指代多个参数,而不是单个参数的重复运用 。第一个下划线代表第一个参数, 第二个下划线代表第二个,第三个……,如此类推。

Scala_函数式编程基础的更多相关文章

  1. 大数据技术之_16_Scala学习_04_函数式编程-基础+面向对象编程-基础

    第五章 函数式编程-基础5.1 函数式编程内容说明5.1.1 函数式编程内容5.1.2 函数式编程授课顺序5.2 函数式编程介绍5.2.1 几个概念的说明5.2.2 方法.函数.函数式编程和面向对象编 ...

  2. javascript函数式编程基础随笔

    JavaScript 作为一种典型的多范式编程语言,这两年随着React\vue的火热,函数式编程的概念也开始流行起来,lodashJS.folktale等多种开源库都使用了函数式的特性. 一.认识函 ...

  3. Scala 中的函数式编程基础(三)

    主要来自 Scala 语言发明人 Martin Odersky 教授的 Coursera 课程 <Functional Programming Principles in Scala>. ...

  4. Scala 中的函数式编程基础(一)

    主要来自 Scala 语言发明人 Martin Odersky 教授的 Coursera 课程 <Functional Programming Principles in Scala>. ...

  5. 3分钟看完Java 8——史上最强Java 8新特性总结之第一篇 函数式编程基础

    目录 · 行为参数化 · Lambda表达式 · 概况 · 函数式接口 · 类型推断 · 使用外层变量 · 方法引用 · 复合Lambda表达式 行为参数化 1. 理解函数式编程要先理解行为参数化. ...

  6. Scala 中的函数式编程基础(二)

    主要来自 Scala 语言发明人 Martin Odersky 教授的 Coursera 课程 <Functional Programming Principles in Scala>. ...

  7. 5. Scala函数式编程的基础

    5.1 函数式编程内容及介绍顺序说明 5.1.1 函数式编程内容 -函数式编程基础 函数定义/声明 函数运行机制 递归(难点,[最短路径,邮差问题,迷宫问题,回溯]) 过程 惰性函数和异常 -函数式编 ...

  8. 让JavaScript回归函数式编程的本质

    JavaScript是一门被误会最深的语言,这话一点不假,我们看下它的发展历史. 1995年,Netscape要推向市场,需要一门脚本语言来配套它.是使用一门已有的语言,还是发明一门新的语言,这也不是 ...

  9. JavaScript中函数式编程中文翻译

    JavaScript 中的函数式编程 原著由 Dan Mantyla 编写 近几年来,随着 Haskell.Scala.Clojure 等学院派原生支持函数式编程的偏门语言越来越受到关注,同时主流的 ...

随机推荐

  1. iOS.Operation-on-ZipFile

    Operation on ZipFile Reference 在Mac OS X和iOS中操作.zip文件(例如创建zip文件, 从zip文件中抽取数据): 1. http://stackoverfl ...

  2. Python爬虫的原理

    简单来说互联网是由一个个站点和网络设备组成的大网,我们通过浏览器访问站点,站点把HTML.JS.CSS代码返回给浏览器,这些代码经过浏览器解析.渲染,将丰富多彩的网页呈现我们眼前: 一.爬虫是什么? ...

  3. Nowcoder 练习赛26E 树上路径 - 树剖

    Description 传送门 给出一个n个点的树,1号节点为根节点,每个点有一个权值 你需要支持以下操作 1.将以u为根的子树内节点(包括u)的权值加val 2.将(u, v)路径上的节点权值加va ...

  4. 动态加载及Servlet容器加载

    动态加载 动态加载是 Servlet 3.0 中的新特性,它可以实现在不重启 Web 应用的情况下加载新的 Web 对象(Servlet.Filter.Listener). 为了实现动态加载的第一种方 ...

  5. Python之路番外(第二篇):PYTHON基本数据类型和小知识点

    一.基础小知识点 1.如果一行代码过长,可以用续行符 \换行书写 例子 if (signal == "red") and \ (car == "moving") ...

  6. python轻量级orm

    python下的orm使用SQLAlchemy比较多,用了一段时间感觉不顺手,主要问题是SQLAlchemy太重,所以自己写了一个orm,实现方式和netsharp类似,oql部分因为代码比较多,没有 ...

  7. jvm 基础

    1. JDK 包含 java 程序设计语言,JVM, Java API类库. java 开发最小环境 2. JRE : Java API类库中java se API 子集和java 虚拟机(HotSp ...

  8. xpath笔记

    参考 1.使用lxml.etree.parse()解析html文件,该方法默认使用的是“XML”解析器,所以如果碰到不规范的html文件时就会解析错误,报错代码如下: lxml.etree.XMLSy ...

  9. 关于 Web Api 2 认证与授权

    认证与授权 认证与授权,Authentication and Authorize,这个是两个不同的事.认证是对访问身份进行确认,如验证用户名和密码,而授权是在认证之后,判断是否具有权限进行某操作,如 ...

  10. 【搜索】棋盘问题(DFS)

    Description 在一个给定形状的棋盘(形状可能是不规则的)上面摆放棋子,棋子没有区别.要求摆放时任意的两个棋子不能放在棋盘中的同一行或者同一列,请编程求解对于给定形状和大小的棋盘,摆放k个棋子 ...