锁定老帖子 主题:有奖竞猜
精华帖 (0) :: 良好帖 (0) :: 新手帖 (0) :: 隐藏帖 (0)
|
|
---|---|
作者 | 正文 |
发表时间:2006-10-02
class Proc def ^ n Proc.new do |*args| (1..n).inject(args){|result, null| result = self.call(*result)} end end end 声明:ITeye文章版权属于作者,受法律保护。没有作者书面许可不得转载。
推荐链接
|
|
返回顶楼 | |
发表时间:2006-10-02
我只测试到这个情况:
(Proc.new{puts 'a'} ^ 3).call 输出: a a a 看这具方法里面new出来的proc的形式,调用应该是这样的: (Proc.new{ |a| puts a; a } ^ 3).call(1) (Proc.new{ |a, b| puts a + b; [a, b] } ^ 3).call(1, 2) (Proc.new{ |a, b, c| puts a + b + c; [a, b, c] } ^ 3).call(1, 2, 3) 看起来没有任何意义。。。 不过倒是让我发现它可以干一件无聊的事: (Proc.new{ |a, b| puts "#{b}, #{a+b}"; [b, a + b] } ^ 8).call(1, 1) 输出: 1, 2 2, 3 3, 5 5, 8 8, 13 13, 21 21, 34 34, 55 => [34, 55] 没发现更多的奥秘了,能不能给下官一个小小的提示? ---------------------------------- 补充一问: (1..n).inject(args){|result, null| result = self.call(*result)} 和 (1..n).inject(args){|result, null| self.call(*result)} 是一个意思吧? |
|
返回顶楼 | |
发表时间:2006-10-02
大致是这样
((Proc.new{|a,b,c| print a,b,c;puts;[c,b,a]} ^ 6).call 1,2,3,4,5,6,7,8,9) ^这个方法会返回一个Proc(简称p)调用传给他的block共n次 这个p接受一个数组参数,调用block的时候,将这个参数展开传给他,并期望这个block返回一个数组a,作为下一次调用block要展开的参数 |
|
返回顶楼 | |
发表时间:2006-10-02
qiezi 写道 补充一问: (1..n).inject(args){|result, null| result = self.call(*result)} 和 (1..n).inject(args){|result, null| self.call(*result)} 是一个意思吧? 嗯,比如 (1..10).inject{|sum,i| sum+=i} 可以写成 (1..10).inject{|sum,i| sum+i} 后一种晦涩一些,一般习惯是前面的写法 基本上那个运算的工作原理不是太难理解,意义可能抽象了点,其实你连fibonacci都搞出来了。。。 |
|
返回顶楼 | |
发表时间:2006-10-02
如果那个Proc接受的参数个数和它返回的个数不一样就失败了。
类似的,还可以定义*,支持像Haskell那种写法f * g class Proc def * p Proc.new do |*args| self[p[*args]] end end end f = lambda { |a| a * a } g = lambda { |b| b + b } h = f * g h.call(3) # => (3 + 3) * (3 + 3) = 36 |
|
返回顶楼 | |
发表时间:2006-10-02
。。这我倒记不太清了。
其实我第一个感觉,这像是一个生成n个用户级线程的东西,但又只能一起call,所以排除。 第二个感觉是一个benchmark工具,不过又不如直接times来得简单。 这个fibonacci是想这家伙的用途的时候想到的,前面已经给出了它的用法,基本上确定它必须:传入n个参数,并且那个proc也要处理n个参数,根据inject用法,这个call也必须返回n个参数,所以fibonacci只是这其中的一个用法。 真要用它来生成fibonacci的话,其实我宁愿写个简单的: irb> (1..5).inject([1,1]){|a,b| a << a[a.length-2] + a.last} => [1, 1, 2, 3, 5, 8, 13] |
|
返回顶楼 | |
发表时间:2006-10-02
Bingo!
既然连f*g(x) = f(g(x))都想到了,应该不会想不到f^n(x) = f(...f(f(x))...)。这个题目好像太数学化了点,罪过。。。主要是我在思考简化一些数组和矩阵运算时的衍生物。 加了点排错的代码: class Proc def ^ n raise "Illegal value of power" unless n >= 1 and n.integer? Proc.new do |*args| raise "Unmatched arguments" unless self.arity == self.call(*args).size (1..n).inject(args){|result, null| result = self.call(*result)} end end end 其实这里面还有个隐蔽的bug嘿嘿。。。 |
|
返回顶楼 | |
发表时间:2006-10-02
果然。改成这样就好看了:
f = lambda{|a,b| [b,a+b]} (f^5).call(1,1) 真服了你们这些FP的脑袋。。 |
|
返回顶楼 | |
发表时间:2006-10-02
lambda {|f| lambda {|h| lambda {|x| f[h[h]][x]}}[lambda {|h| lambda {|x| f[h[h]][x]}}]}[lambda {|f| lambda {|x| x<2 ? 1 : f[x-1] + f[x-2]}}][5] # => 8 显示不下了-_- lambda {|f| lambda {|h| lambda {|x| f[h[h]][x]}}[lambda {|h| lambda {|x| f[h[h]][x]}}]}\ [lambda {|f| lambda {|x| x<2 ? 1 : f[x-1] + f[x-2]}}][5] # => 8 |
|
返回顶楼 | |
发表时间:2006-10-02
qiezi 写道 果然。改成这样就好看了:
f = lambda{|a,b| [b,a+b]} (f^5).call(1,1) 真服了你们这些FP的脑袋。。 其实还有不少例子,比如求质数: prime = lambda{|a,*b| print a,' '; b.delete_if{|x|x%a == 0}} (prime^10).call *(2..100) #=> 2 3 5 7 11 13 17 19 23 29 诸如此类在同一数据结构上做重复操作的都可以这么简化 |
|
返回顶楼 | |