Thinking in scala (5)----高阶函数*
高阶函数是函数式编程里面一个非常重要的特色,所谓的高阶函数,就是以其它函数作为参数的函数。
下面以一个小例子演示Scala的高阶函数特性,非常有意思,也非常强大。
首先看这么一个程序:
code1:
object higherorderfuntion{
def sum1(a:Int,b:Int):Int=
if(a>b) 0
else a+sum1(a+1,b)
def sum2(a:Int,b:Int):Int=
if(a>b) 0
else cube(a)+sum2(a+1,b)
def sum3(a:Int,b:Int):Int=
if(a>b) 0
else fac(a)+sum3(a+1,b)
def cube(a:Int):Int=a*a*a
def fac(a:Int):Int=
if (a==0) 1
else a*fac(a-1)
def main(args:Array[String])={
println(sum1(1,3))
println(sum2(1,3))
println(sum3(1,3))
}
}
上面这个例子“没有”用到高阶函数,sum1是计算a+(a+1)+(a+2)+...+(b), sum2是计算a^3+(a+1)^3+(a+2)^3+...+b^3,
sum3是计算a!+(a+1)!+(a+2)!+...+b!。分析sum1,sum2,sum3的代码,很容易发现这三个函数有着相似的“Pattern”,
抛开函数名不论,这三个函数唯一的区别在于,在 else语句中对a的处理:a, cube(a) , fac(a). 那么来看,在函数式
编程里面,是如何非常精彩的利用这个“Pattern”来使得代码更加精简的:
code2:
object higherorderfuntion{
def sum(f:Int=>Int,a:Int,b:Int):Int=
if(a>b) 0
else f(a)+sum(f,a+1,b)
def sumInt(a:Int,b:Int):Int = sum(id,a,b)
def sumCube(a:Int,b:Int):Int = sum(cube,a,b)
def sumFac(a:Int,b:Int):Int = sum(fac,a,b)
def id(a:Int) = a
def cube(a:Int):Int=a*a*a
def fac(a:Int):Int=
if (a==0) 1
else a*fac(a-1)
def main(args:Array[String])={
println(sumInt(1,3))
println(sumCube(1,3))
println(sumFac(1,3))
}
}
重头戏来了,我们来看sum函数的实现:
def sum(f:Int=>Int,a:Int,b:Int):Int=
if(a>b) 0
else f(a)+sum(f,a+1,b)
sum函数接收三个参数,第二个和第三个参数分别是: a:Int 和 b:Int, 这和第一个例子中是一样的。
比较令人费解的是sum函数的第一个参数:f:Int=>Int
这个是什么意思呢?意思是sum函数接收一个名字叫做 “f” 的函数作为参数,而 Int=>Int 是对f的说明:
=>左边的Int是说:函数f接收一个Int类型的参数,
=>右边的Int是说:函数f的返回值是Int类型的。
好了,那么既然sum函数接收函数f作为一个参数,那么sum就可以利用f了,事实也是这样的,看sum
函数的else语句就知道了:f(a)
然后再看sumInt、sumCube和sumFac三个函数的定义:
def sumInt(a:Int,b:Int):Int = sum(id,a,b) def sumCube(a:Int,b:Int):Int = sum(cube,a,b) def sumFac(a:Int,b:Int):Int = sum(fac,a,b)
以sumCube函数为例吧,它在定义的时候,把cube函数作为参数,传递给sum函数,
而我们看cube函数的定义:
def cube(a:Int):Int=a*a*a
发现,cube函数接收一个Int作为参数,并且返回一个Int,也就是说cube函数是符合sum函数的第一个
参数: f:Int=>Int 的
是不是很精彩呢?
事实上上述代码还可以进一步简化,因为我们观察到id函数和cube函数的功能非常简单,不需要单独作为
一个函数出现,进一步简化后的代码如下:
code3:
object higherorderfuntion{
def sum(f:Int=>Int,a:Int,b:Int):Int=
if(a>b) 0
else f(a)+sum(f,a+1,b)
def sumInt(a:Int,b:Int):Int = sum(x=>x,a,b)
def sumCube(a:Int,b:Int):Int = sum(x=>x*x*x,a,b)
def sumFac(a:Int,b:Int):Int = sum(fac,a,b)
def fac(a:Int):Int=
if (a==0) 1
else a*fac(a-1)
def main(args:Array[String])={
println(sumInt(1,3))
println(sumCube(1,3))
println(sumFac(1,3))
}
}
code3和code2相比,去掉了id函数和cube这两个函数,并且sumInt和sumCube函数的声明也发生了
一点变化:
def sumInt(a:Int,b:Int):Int = sum(x=>x,a,b) def sumCube(a:Int,b:Int):Int = sum(x=>x*x*x,a,b)
像: x=>x 和 x=>x*x*x 这样的东西是什么呢?在函数式编程里面,这被叫做“function literal”,又称“匿名函数”,
说白了,这也是函数的一种表达方式,只是这个函数没有名字罢了。
code3其实也还有点小毛病,那就是sum函数和fac函数都不是“尾递归”,所以呢,把它们改成尾递归如下:
code4:
object higherorderfuntion{
def sum(f:Int=>Int,a:Int,b:Int):Int={
def loop(a:Int,acc:Int):Int=
if(a>b) acc
else loop(a+1,f(a)+acc)
loop(a,0)
}
def sumInt(a:Int,b:Int):Int = sum(x=>x,a,b)
def sumCube(a:Int,b:Int):Int = sum(x=>x*x*x,a,b)
def sumFac(a:Int,b:Int):Int = sum(fac,a,b)
def fac(a:Int):Int=
{
def loop(a:Int,acc:Int):Int=
if(a==0) acc
else loop(a-1,a*acc)
loop(a,1)
}
def main(args:Array[String])={
println(sumInt(1,4))
println(sumCube(1,4))
println(sumFac(1,4))
}
}
希望把这个事儿说明白了...........
For any questions, feel free to contact me via email or QQ.
Thinking in scala (5)----高阶函数*的更多相关文章
- Thinking in scala (6)----高阶函数----返回一个函数
在Thinking in scala (5)----高阶函数* 里面,我们演示了如何把一个函数作为参数传递给另外一个函数. 在本文里面,我们来演示函数式编程另外一个重要的特性:返回一个函数.首先来看这 ...
- scala面向对象.高阶函数,柯里化,Actor编程简介
1.定义一个类 class Person{ //用val修饰的变量是只读属性,有getter但是没有setter val id ="111" //用var修饰的变量既有getter ...
- Scala中的构造器和高阶函数
构造器 在定义类时可以定义主构造器.主构造器可以同时声明字段. /** * 主构造器 * @author Administrator */ //在scala中,类和方法交织在一起 class Test ...
- Scala - 快速学习08 - 函数式编程:高阶函数
函数式编程的崛起 函数式编程中的“值不可变性”避免了对公共的可变状态进行同步访问控制的复杂问题,能够较好满足分布式并行编程的需求,适应大数据时代的到来. 函数是第一等公民 可以作为实参传递给另外一个函 ...
- spark快速开发之scala基础之5高阶函数,偏函数,闭包
高阶函数 高阶函数就是将函数作为参数或者返回值的函数. object function { def main(args: Array[String]): Unit = { println(test(f ...
- scala高阶函数类型推断什么时候失效?
class TypeInfer(self: Int, other: Int) { def test(num: Int, word: String, fun1: (Int, Int) => Int ...
- Scala高阶函数
1.作为参数的函数 函数可以作为一个参数传入到一个方法当中去 def main(args: Array[String]): Unit = { val myFunc1 =(x:Int) =>{ x ...
- scala中的高阶函数
版权申明:转载请注明出处. 文章来源:http://bigdataer.net/?p=332 排版乱?请移步原文获得更好阅读体验 1.scala中的函数 scala是一门面向对象和函数式编程相结合的语 ...
- Scala高阶函数与泛型
1. Scala中的函数 在Scala中,函数是“头等公民”,就和数字一样.可以在变量中存放函数,即:将函数作为变量的值(值函数). 2. scala中的匿名函数,即没有函数名称的函数,匿名函数常作为 ...
随机推荐
- HDU-1301 Jungle Roads(最小生成树[Prim])
Jungle Roads Time Limit : 2000/1000ms (Java/Other) Memory Limit : 65536/32768K (Java/Other) Total ...
- Servlet简介及使用
javaweb学习总结(五)——Servlet开发(一) 一.Servlet简介 Servlet是sun公司提供的一门用于开发动态web资源的技术. Sun公司在其API中提供了一个servlet接口 ...
- 滑轮关节(b2PulleyJoint)
package{ import Box2D.Collision.b2AABB; import Box2D.Collision.b2RayCastInput; import Box2D.Collisio ...
- KNN邻近分类算法
K邻近(k-Nearest Neighbor,KNN)分类算法是最简单的机器学习算法了.它采用测量不同特征值之间的距离方法进行分类.它的思想很简单:计算一个点A与其他所有点之间的距离,取出与该点最近的 ...
- Redis SAVE 命令 返回 ERR
今天使用redis-cli客户端中执行SAVE命令返回 (error) ERR Baidu找不到答案,去Google找一下 应该是redis-server服务没有root权限 然后sudo kill ...
- centos 7.1搭建docker本地私有仓库返回500错误
之前有一篇写到在ubuntu14.04系统上安装私有仓库,遇到了两个问题,本次在centos7上遇到了另外一个问题. 安装完仓库并运行registry镜像之后发现push和pull操作都会返回一个50 ...
- android 画图之setXfermode .
setXfermode 设置两张图片相交时的模式 我们知道 在正常的情况下,在已有的图像上绘图将会在其上面添加一层新的形状. 如果新的Paint是完全不透明的,那么它将完全遮挡住下面的Paint: 而 ...
- 转:web_submit_data函数
web_submit_data函数处理无状态或者上下文无关的表单提交.它用来生成表单的GET或POST请求,这些请求与Form自动生成的请求是一样的.发送这些请求时不需要表单上下文. 函数语法:Int ...
- PAT (Advanced Level) 1049. Counting Ones (30)
数位DP.dp[i][j]表示i位,最高位为j的情况下总共有多少1. #include<iostream> #include<cstring> #include<cmat ...
- Linux服务器自动备份压缩MySQL数据库的实用方法
<?php$server = 'localhost'; $link = mysql_connect($server, 'root', 'haven'); $result = mysql_quer ...