在ruby中递归地将块传递给方法

Posted

技术标签:

【中文标题】在ruby中递归地将块传递给方法【英文标题】:Passing a block to a method recursively in ruby 【发布时间】:2014-11-26 00:12:52 【问题描述】:
def bubble_sort_by nums
  do_it_again = false
  nums[0...-1].each_with_index do |item, index|
    if yield(nums[index], nums[index + 1]) > 0
      nums[index], nums[index + 1] = nums[index + 1], nums[index]
      do_it_again = true
    end
  end
  bubble_sort_by nums if do_it_again
  nums
end

bubble_sort_by(["hi","hello","hey"]) do |left,right|
  right.length - left.length
end

程序基于块进行冒泡排序。在这种情况下,块按长度排序。所以,我得到一个本地跳转错误。花了我一点,但我想通了。当我递归调用该方法时,我没有给它阻止。但是我该怎么做呢?

【问题讨论】:

【参考方案1】:

yield + block_given 解决方案是一次性的,如果没有 显式 传递 &block,您不能在递归调用中使用它>Proc(一个闭包)作为一个参数,尽管它在经验上会产生更好的性能。

这是一些用于递归调用的虚拟代码(地图函数的自定义版本)

将 Proc 作为普通对象传递

def mymap(xs, f)
  if xs.empty? 
    [] 
  else
    hd, *tl = xs
    [f.call(hd)]+ mymap(tl,f)
  end  
end
mymap([1,2,3], lambda |x| x*2)

传递可选的 &block

def mymmap(xs, &block)
  if xs.empty? 
    [] 
  else
    hd, *tl = xs
    [block.call(hd)]+ mymmap(tl, &block) #here yield(hd) will do as well
  end  
end
mymmap([1,2,3])|x| 2*x #notice the calling manner is different from the above 

【讨论】:

【参考方案2】:

传递块是可选的,但在这里,正如您已经理解的那样,您需要它来进行递归调用。您只需将块作为附加参数传递:

def bubble_sort_by nums, &comparator
  # ...
  bubble_sort_by nums, &comparator if do_it_again
  # ...
end

【讨论】:

以上是关于在ruby中递归地将块传递给方法的主要内容,如果未能解决你的问题,请参考以下文章

将块传递给延迟作业

如何将多个参数作为数组传递给 ruby​​ 方法?

在 C# 中如何正确地将 int 数组传递给函数

Ruby - 将方法作为回调传递给 DLL 库

将块中异步接收的 JSON 对象传递给实例变量 - 在 iOS6 上使用 AFNetworking

如何正确地将页面属性传递给局部视图?