在PHP中模拟ruby的inject()行为

qua*_*oup 3 php closures

这里的这个问题来看,我正在编写一个枚举包装器,以便有一些方法可以与lambdas一起使用,以在某种程度上模拟ruby在枚举中对块的使用.

class enum {
    public $arr;

    function __construct($array) {
        $this->arr = $array;
    }

    function each($lambda) {
        array_walk($this->arr, $lambda);
    }

    function find_all($lambda) {
        return array_filter($this->arr, $lambda);
    }

    function inject($lambda, $initial=null) { 
        if ($initial == null) { 
            $first = array_shift($this->arr); 
            $result = array_reduce($this->arr, $lambda, $first); 
            array_unshift($this->arr, $first); 

            return $result; 
        } else { 
            return array_reduce($this->arr, $lambda, $initial); 
        } 
    } 

}


$list = new enum(array(-1, 3, 4, 5, -7));
$list->each(function($a) { print $a . "\n";});

// in PHP you can also assign a closure to a variable 
$pos = function($a) { return ($a < 0) ? false : true;};
$positives = $list->find_all($pos);
Run Code Online (Sandbox Code Playgroud)

现在,我怎样才能尽可能优雅地实现inject()


编辑:如上所述实现的方法.用法示例:

// inject() examples 
$list = new enum(range(5, 10)); 

$sum = $list->inject(function($sum, $n) { return $sum+$n; }); 
$product = $list->inject(function($acc, $n) { return $acc*$n; }, 1); 

$list = new enum(array('cat', 'sheep', 'bear')); 
$longest = $list->inject(function($memo, $word) { 
        return (strlen($memo) > strlen($word)) ? $memo : $word; } 
    ); 
Run Code Online (Sandbox Code Playgroud)

Art*_*cto 5

我不熟悉Ruby,但从描述来看,它似乎相似array_reduce.

mixed array_reduce ( array $input , callback $function [, mixed $initial = NULL ] )

array_reduce() 迭代地将函数函数应用于数组输入的元素,以便将数组减少为单个值.

除了"减少"之外,这种操作有时也称为"折叠" ; 在Mathematica:

Fold[f, init, {a, b, c, d}] == f[f[f[f[init, a], b], c], d]
Run Code Online (Sandbox Code Playgroud)

第二种形式使用集合的第一个元素作为初始值(并在迭代时跳过该元素).

第二种形式可以这样实现:

//$arr is the initial array
$first = array_shift($arr);
$result = array_reduce($arr, $callback, $first);
Run Code Online (Sandbox Code Playgroud)

回应Mladen

PHP中的数组函数不能以这种方式使用,因为它们只能用于数组,而不能用于任意对象.

这里有几个选项:

  • 您可以在将对象传递给数组之前将其转换为数组array_reduce.实际上,这没有多大价值,因为转换包括创建一个以对象属性为元素的数组.此行为只能在内部更改(编写本机扩展).
  • 您可以让所有对象使用一种方法实现一个接口,该方法toArray必须先被调用才能传递给它array_reduce.也不是一个好主意.
  • 您可以实现array_reduce适用于任何Traversable对象的版本.这很容易做到,但Traversable由于数组不是对象,因此无法在函数声明中添加类型提示.有了这样的提示,每个数组都必须ArrayIterator在函数调用之前封装在一个对象中.