这是一个有趣的讨论。我认为@flodel的例子很好。然而,我认为这说明了我的观点(@koshke在评论中提到了这一点),当您使用命令式而不是函数式编码风格时,返回是有意义的。
我不想赘述这一点,但是我会像这样重写foo:
foo = function() ifelse(a,a,b)
函数式风格避免了状态更改,比如存储输出值。在这种风格下,回归是不合适的;Foo看起来更像一个数学函数。
我同意@flodel的观点:在酒吧中使用复杂的布尔变量系统会不太清楚,而且当你返回时毫无意义。使bar如此易于返回语句的原因是它是用命令式风格编写的。实际上,布尔变量表示以函数式风格避免的“状态”变化。
以函数式的方式重写bar是非常困难的,因为它只是伪代码,但其思想是这样的:
e_func <- function() do_stuff
d_func <- function() ifelse(any(sapply(seq(d),e_func)),2,3)
b_func <- function() {
do_stuff
ifelse(c,1,sapply(seq(b),d_func))
}
bar <- function () {
do_stuff
sapply(seq(a),b_func) # Not exactly correct, but illustrates the idea.
}
while循环是最难重写的,因为它是由变量的状态更改控制的。
调用return所造成的速度损失可以忽略不计,但是通过避免返回并以函数式风格重写所获得的效率通常是巨大的。告诉新用户停止使用return可能不会有帮助,但引导他们使用函数式风格会有回报。
@Paul return在命令式中是必要的,因为您经常希望在循环中的不同位置退出函数。函数式风格不使用循环,因此不需要return。在纯函数式风格中,最终调用几乎总是所需的返回值。
在Python中,函数需要返回语句。然而,如果你用函数式风格编写函数,你可能只有一个return语句:在函数的末尾。
使用另一篇StackOverflow帖子中的示例,假设我们想要一个函数,如果给定x中的所有值都是奇数长度,则该函数返回TRUE。我们可以使用两种样式:
# Procedural / Imperative
allOdd = function(x) {
for (i in x) if (length(i) %% 2 == 0) return (FALSE)
return (TRUE)
}
# Functional
allOdd = function(x)
all(length(x) %% 2 == 1)
在函数式样式中,返回的值自然落在函数的末尾。它看起来更像一个数学函数。
@GSee ?ifelse中列出的警告确实很有趣,但我不认为他们试图劝阻使用该功能。事实上,ifelse具有自动向量化函数的优点。例如,考虑稍微修改过的foo:
foo = function(a) { # Note that it now has an argument
if(a) {
return(a)
} else {
return(b)
}
}
当length(a)为1时,此函数工作正常。但如果你用ifelse重写foo
foo = function (a) ifelse(a,a,b)
现在foo适用于a的任何长度,事实上,它甚至适用于a是一个矩阵。返回与test形状相同的值是一个有助于向量化的特性,不是问题。