我正在使用一个bash脚本,我想执行一个函数打印返回值:

function fun1(){
  return 34
}
function fun2(){
  local res=$(fun1)
  echo $res
}

当我执行fun2时,它不会打印“34”。为什么会这样呢?


当前回答

其他答案的问题是,它们要么使用全局变量,当几个函数在一个调用链中时,它可以被覆盖,要么使用echo,这意味着你的函数不能输出诊断信息(你会忘记你的函数是这样做的,而“结果”,即返回值,将包含比调用者期望的更多的信息,导致奇怪的错误),要么使用eval,这太沉重了。

正确的方法是将顶级的东西放在一个函数中,并使用带有Bash动态作用域规则的局部函数。例子:

func1()
{
    ret_val=hi
}

func2()
{
    ret_val=bye
}

func3()
{
    local ret_val=nothing
    echo $ret_val
    func1
    echo $ret_val
    func2
    echo $ret_val
}

func3

这个输出

nothing
hi
bye

动态作用域意味着ret_val指向不同的对象,这取决于调用者!这与大多数编程语言使用的词法作用域不同。这实际上是一个有文档记录的特性,只是很容易被忽略,也没有很好地解释。以下是它的文档(重点是我的):

函数的局部变量可以用局部变量声明 内装式。这些变量仅对函数和对象可见 它调用的命令。

对于具有C、c++、Python、Java、c#或JavaScript背景的人来说,这可能是最大的障碍:bash中的函数不是函数,它们是命令,并且它们的行为是这样的:它们可以输出到stdout/stderr,它们可以通过管道输入/输出,并且它们可以返回退出代码。基本上,在脚本中定义命令和创建可从命令行调用的可执行文件之间没有任何区别。

所以不要像这样写脚本:

Top-level code
Bunch of functions
More top-level code

这样写:

# Define your main, containing all top-level code
main()
Bunch of functions
# Call main
main

main()将ret_val声明为本地,所有其他函数通过ret_val返回值。

请参见Unix和Linux问题Shell函数中局部变量的作用域。

另一个,根据情况可能更好的解决方案,是由你发布的。使用本地-n的Teck。

其他回答

另一种实现方法是名称引用(需要Bash 4.3+)。

function example {
  local -n VAR=$1
  VAR=foo
}

example RESULT
echo $RESULT

我能想到的最简单的方法是在方法体中像这样使用echo

get_greeting() {
  echo "Hello there, $1!"
}

STRING_VAR=$(get_greeting "General Kenobi")
echo $STRING_VAR
# Outputs: Hello there, General Kenobi!

如果在定义函数的脚本中运行,我喜欢做以下事情:

POINTER= # Used for function return values

my_function() {
    # Do stuff
    POINTER="my_function_return"
}

my_other_function() {
    # Do stuff
    POINTER="my_other_function_return"
}

my_function
RESULT="$POINTER"

my_other_function
RESULT="$POINTER"

我喜欢这样,因为我可以在函数中包含echo语句

my_function() {
    echo "-> my_function()"
    # Do stuff
    POINTER="my_function_return"
    echo "<- my_function. $POINTER"
}

其他答案的问题是,它们要么使用全局变量,当几个函数在一个调用链中时,它可以被覆盖,要么使用echo,这意味着你的函数不能输出诊断信息(你会忘记你的函数是这样做的,而“结果”,即返回值,将包含比调用者期望的更多的信息,导致奇怪的错误),要么使用eval,这太沉重了。

正确的方法是将顶级的东西放在一个函数中,并使用带有Bash动态作用域规则的局部函数。例子:

func1()
{
    ret_val=hi
}

func2()
{
    ret_val=bye
}

func3()
{
    local ret_val=nothing
    echo $ret_val
    func1
    echo $ret_val
    func2
    echo $ret_val
}

func3

这个输出

nothing
hi
bye

动态作用域意味着ret_val指向不同的对象,这取决于调用者!这与大多数编程语言使用的词法作用域不同。这实际上是一个有文档记录的特性,只是很容易被忽略,也没有很好地解释。以下是它的文档(重点是我的):

函数的局部变量可以用局部变量声明 内装式。这些变量仅对函数和对象可见 它调用的命令。

对于具有C、c++、Python、Java、c#或JavaScript背景的人来说,这可能是最大的障碍:bash中的函数不是函数,它们是命令,并且它们的行为是这样的:它们可以输出到stdout/stderr,它们可以通过管道输入/输出,并且它们可以返回退出代码。基本上,在脚本中定义命令和创建可从命令行调用的可执行文件之间没有任何区别。

所以不要像这样写脚本:

Top-level code
Bunch of functions
More top-level code

这样写:

# Define your main, containing all top-level code
main()
Bunch of functions
# Call main
main

main()将ret_val声明为本地,所有其他函数通过ret_val返回值。

请参见Unix和Linux问题Shell函数中局部变量的作用域。

另一个,根据情况可能更好的解决方案,是由你发布的。使用本地-n的Teck。

$(…)捕获其中包含的命令发送到标准输出的文本。返回不输出到标准输出。$ ?包含最后一个命令的结果代码。

fun1 (){
  return 34
}

fun2 (){
  fun1
  local res=$?
  echo $res
}