Ramda函数式编程之PHP
0x00 何为函数式编程
网上已经有好多详细的接受了,我认为比较重要的有:
- 函数是“第一等公民”,即函数和其它数据类型一样处于平等地位
- 使用“表达式”(指一个单纯的运算过程,总是有返回值),而不是“语句”(执行操作,没有返回值)
- 没有”副作用“,即不修改外部值
0x01 开始函数式编程
在此之前,请先了解PHP中的匿名函数和闭包,可以参考我写得博客
函数式编程有两个最基本的运算:合成和柯里化。
函数合成
函数合成,即把多个函数的运算合成一个函数,如
A=f(x)
B=g(x)
C=f(g(x))
那么C即是A和B的合成。
用代码表示为:
$compose = function ($f,$g){
return function ($x) use($f,$g){ //这里返回一个函数的函数,即高阶函数
return $f($g($x));
};
};
function addTen($a){
return $a+10;
}
function subOne($a){
return $a-1;
}
$z = $compose('addTen','subOne');//如果使用 $addOne = function(){}的形式,可以直接传变量
echo $z(5);// 14
要求合成的函数也是个纯函数,如果不是纯函数,那么结果不一致,怎么合成呢?
compose返回一个高阶函数,当给合成的这个函数传值时,变回在高阶函数内部调用之前保存的函数。
柯里化
可以看到如果这里传入的函数参数有多个,那么上面的合成函数就失效了。
这里就要请出另外一个函数式编程使用到的另外一个大神了,柯里化。“所谓"柯里化",就是把一个多参数的函数,转化为单参数函数"。
//柯里化之前
function add($a,$b){
return $a+$b;
}
add(1, 2); // 3
// 柯里化之后
function addX($b) {
return function ($a) use($b) {
return $a + $b;
};
}
$addTwo = addX(2);
$addTwo(1);//3
PHP7以下直接调用addX(2)(1),会报错,所以上面使用了中间变量$addTwo。
Parse error: syntax error, unexpected '('
在PHP7以上完善了一致变量语法,而且PHP7速度更快,强烈建议使用PHP7。
通用柯里化,柯里化很美好,然而我们不可能为每一个函数写一遍,那么有没有包装函数,可以把普通的函数改些为柯里化后的函数呢?
代码如下:(摘自:pramda)
function curry2($callable)
{
return function () use ($callable) {
$args = func_get_args();
switch (func_num_args()) {
case 0:
throw new \Exception("Invalid number of arguments");
break;
case 1:
return function ($b) use ($args, $callable) {
return call_user_func_array($callable, [$args[0], $b]);
};
break;
case 2:
return call_user_func_array($callable, $args);
break;
default:
// Why? To support passing curried functions as parameters to functions that pass more that 2 parameters, like reduce
return call_user_func_array($callable, [$args[0], $args[1]]);
break;
}
};
}
function add($a,$b){
return $a+$b;
}
$addCurry = curry2('add');
$addTwo = $addCurry(2);
$addTwo(1);//3
说明,curry2返回一个闭包(如上面的$addCurry),当这个闭包被调用时会通过func_get_args动态获取参数,
以及func_num_args动态获取参数个数。curry2函如其名,可以给把参数个数为两个函数柯里化。于是在闭包里,我们看到,
在对参数个数进行判断,当参数个数为1时,则生成新的闭包(如上面的$addTwo),新的闭包里保存原函数以及整个参数,
当新闭包被调用时,则调用call_user_func_array传入原函数、保存的参数、新参数,获取了想要的结果。
扩展,函数式编程还有另外一个重要的概念,函子(即带有map方法的类),更多内容可以看阮老师的这两篇文章,我就不详叙了。
平常我们自己使用的函数,如果符合函数式编程的思想,也可以柯里化。当然对于更多参数的函数得运用更高阶的curryN来柯里化。
这些已经有人造好轮子了,下面开始进入正题了。
0x02 Ramda
这个Ramda实际上是函数式编程中的Pointfree风格。
在Ramda里,数据一律放在最后一个参数,理念是"function first,data last"。
比如
//例1
function map(){
$args = func_get_args();
$n = func_num_args();
$callable = $args[$n-1];
unset($args[$n-1]);
$res = [];
foreach ($args as $v){
if(is_array($v)){
foreach ($v as $i){
$res[] = call_user_func($callable,$i);
}
}else{
$res[] = call_user_func($callable,$v);
}
}
return $res;
}
map(1,2,'square');//1,4
map([1,2],'square');// 1,4
//例2
function square($v)
{
return($v*$v);
}
array_map("square",[1,2]); //1 ,4
上面的代码,例1就不是Ramda风格,而例2则是Ramda风格。
既然有人造好轮子了,那么我们直接用就好啦,下面请出主角,pramda,Ramda风格的PHP函数式编程库。
安装
composer require kapolos/pramda
如果出现
[InvalidArgumentException]
Could not find package kapolos/pramda.
可以在composer.json里加入 "kapolos/pramda":"dev-master"
示例:
$before = [1,2,3,4,5];
$after = P::map(function($num) {
return $num * 2;
}, $before);
P::toArray($after); //=> [2,4,6,8,10]
$addOne = P::add(1);
$divTen = P::divide(10); //10是被除数
$fn1 = P::compose($addOne,$divTen); //compose从右往左
$fn2 = P::pipe($addOne,$divTen);//pipe从左往右
echo $fn1(1); //11
echo "\n";
echo $fn2(1); //5
不足之处,pramda不支持占位符,另外curry函数最多只支持3个参数。
另外有也有两个函数式编程库,functional-php和dash可惜不是Ramda风格的。
正如阮老师所提到的
学习函数式编程,实际上就是学习函子的各种运算。
如果想了解更多,可以继续阅读阮老师的这两篇文章。
Ramda函数式编程之PHP的更多相关文章
- Python函数式编程之map()
Python函数式编程之map() Python中map().filter().reduce()这三个都是应用于序列的内置函数. 格式: map(func, seq1[, seq2,…]) 第一个参数 ...
- python3 第二十一章 - 函数式编程之return函数和闭包
我们来实现一个可变参数的求和.通常情况下,求和的函数是这样定义的: def calc_sum(*args): ax = 0 for n in args: ax = ax + n return ax 但 ...
- python3 第二十章 - 函数式编程之Higher-order function(高阶函数)
什么是高阶函数?把函数作为参数传入或把函数做为结果值返回,这样的函数称为高阶函数,函数式编程就是指这种高度抽象的编程范式.函数式编程的特点: 函数本身可以赋值给变量,赋值后变量为函数: 允许将函数本身 ...
- 函数式编程之foldLeftViaFoldRight
问题来自 Scala 函数式编程 一书的习题, 让我很困扰, 感觉函数式编程有点神学的感觉.后面看懂之后, 又觉得函数式编程所提供的高阶抽象是多么的强大. 这个问题让我发呆了好久, 现在把自己形成的想 ...
- Python函数式编程之lambda表达式
一:匿名函数的定义 lambda parameter_list: expression 二:三元表达式 条件为真时返回的结果 if 条件判断 else 条件为假的时候返回的结果 三:map map(f ...
- 函数式编程之-bind函数
Bind函数 Bind函数在函数式编程中是如此重要,以至于函数式编程语言会为bind函数设计语法糖.另一个角度Bind函数非常难以理解,几乎很少有人能通过简单的描述说明白bind函数的由来及原理. 这 ...
- 函数式编程之-F#类型系统
在深入到函数式编程思想之前,了解函数式独有的类型是非常有必要的.函数式类型跟OO语言中的数据结构截然不同,这也导致使用函数式编程语言来解决问题的思路跟OO的思路有明显的区别. 什么是类型?类型在编程语 ...
- 函数式编程之-Partial application
上一篇关于Currying的介绍,我们提到F#是如何做Currying变换的: let addWithThreeParameters x y z = x + y + z let intermediat ...
- 函数式编程之-Currying
这个系列涉及到了F#这门语言,也许有的人觉得这样的语言遥不可及,的确我几乎花了2-3年的时间去了解他:也许有人觉得学习这样的冷门语言没有必要,我也赞同,那么我为什么要花时间去学习呢?作为一门在Tiob ...
随机推荐
- 第一次亲密接触MSF
第一次亲密接触MSF Metasploit Framework介绍 Metasploit是一款开源安全漏洞检测工具,附带数百个已知的软件漏洞,并保持频繁更新.被安全社区冠以“可以黑掉整个宇宙”之名的强 ...
- 移动端H5页面返回并且刷新页面(BFcache)
项目中的需求:点击浏览器中的返回按钮,要让页面重新加载资源.因为这部分的资源每次去加载的内容都不一样,如果返回的时候,还是看到原先的内容,那做这个内容块的意义就很小了:而如果用户看完了这部分内容,再返 ...
- JavaScript函数使用技巧
JavaScript中的函数是整个语言中最有趣的一部分,它们强大而且灵活.接下来,我们来讨论JavaScript中函数的一些常用技巧: 一.函数绑定 函数绑定是指创建一个函数,可以在特定的this环境 ...
- 安卓 logcat设置 Android logcat Settings
安卓 logcat设置 Android logcat Settings 作者:韩梦飞沙 Author:han_meng_fei_sha 邮箱:313134555@qq.com E-mail: 3131 ...
- Python3练习题系列(08)——代码阅读方法及字典跳转表理解
问题:分析下面代码 cities['_find'] = find_city city_found = cities['_find'](cities, state) 分析过程: 一个函数也可以作为一个变 ...
- [HDU5968]异或密码
[HDU5968]异或密码 题目大意: 数据共\(T(T\le100)\)组.每组给定一个长度为\(n(n\le100)\)的非负整数序列\(A(A_i\le1024)\),\(m(m\le100)\ ...
- android圆角功能,非常好用,可以用在图片,视频,gif等上面
使用方式:直接在xml中使用即可. <com.base.baseview.RoundLayout android:id="@+id/example_view" android ...
- 20172302 《Java软件结构与数据结构》第八周学习总结
2018年学习总结博客总目录:第一周 第二周 第三周 第四周 第五周 第六周 第七周 第八周 教材学习内容总结 第十二章 优先队列与堆 1.堆(heap)是具有两个附加属性的一棵二叉树: (1)它是一 ...
- BZOJ2759一个动态树好题 LCT
题如其名啊 昨天晚上写了一发忘保存 只好今天又码一遍了 将题目中怕$p[i]$看做$i$的$father$ 可以发现每个联通块都是一个基环树 我们对每个基环删掉环上一条边 就可以得到一个森林了 可以用 ...
- 来自极客头条的 35 个 Java 代码性能优化总结
前言 代码优化,一个很重要的课题.可能有些人觉得没用,一些细小的地方有什么好修改的,改与不改对于代码的运行效率有什么影响呢?这个问题我是这么考虑的,就像大海里面的鲸鱼一样,它吃一条小虾米有用吗?没用, ...