Bash函数中的返回语句和退出语句在退出代码方面有什么区别?


当前回答

Exit终止当前进程;不管有没有退出码,都应该把它看作一个系统,而不是一个程序函数。注意,当使用源时,exit将结束shell。但是,在运行时,它将退出脚本。 从函数返回,返回到调用后的指令,带或不带返回码。Return是可选的,在函数末尾是隐式的。Return只能在函数内部使用。

我想补充一点,在获取源代码的同时,不杀死shell而从函数中退出脚本并不容易。我认为,“测试”脚本上的示例更好:

#!/bin/bash
function die(){
   echo ${1:=Something terrible wrong happen}
   #... clean your trash
   exit 1
}

[ -f /whatever/ ] || die "whatever is not available"
# Now we can proceed
echo "continue"

做以下事情:

user$ ./test
Whatever is not available
user$

测试后,外壳会闭合。

user$ . ./test
Whatever is not available

只有测试将完成,并显示提示。

解决方案是将潜在的过程包含在(和)中:

#!/bin/bash
function die(){
   echo $(1:=Something terrible wrong happen)
   #... Clean your trash
   exit 1
}

( # Added        
    [ -f /whatever/ ] || die "whatever is not available"
    # Now we can proceed
    echo "continue"
) # Added

现在,在这两种情况下,只有test将退出。

其他回答

从man bash on return [n];

使函数停止执行并将n指定的值返回给调用者。如果省略n,返回状态为函数体中最后执行的命令。

... 退出[n]:

使shell以状态n退出。如果省略n,退出状态为最后执行的命令。EXIT上的trap在shell终止之前执行。

编辑:

根据你对问题的编辑,关于退出码,return和退出码没有关系。退出码用于应用程序/脚本,而不是函数。因此,在这方面,唯一设置脚本退出码的关键字(调用程序使用$?Shell变量)是退出。

编辑2:

我提到退出的最后一句话引起了一些评论。这样做是为了区分return和exit,以便理解OP,事实上,在程序/shell脚本的任何给定点上,exit是使用调用进程的退出代码结束脚本的唯一方法。

在shell中执行的每个命令都会产生一个本地“退出码”:它设置$?变量,可以与if, &&和其他操作符一起使用,有条件地执行其他命令。

这些退出码(以及$?变量)将在每次命令执行时重置。

顺便提一下,脚本执行的最后一个命令的退出码被用作调用进程看到的脚本本身的退出码。

最后,函数在被调用时,作为关于退出代码的shell命令。函数的退出码(在函数内)是通过使用return设置的。因此,当在函数中运行return 0时,函数执行将终止,并给出0的退出码。

OP的问题是: BASH函数中的返回语句和退出语句在退出代码方面有什么区别?

首先,需要做一些澄清:

A (return|exit) statement is not required to terminate execution of a (function|shell). A (function|shell) will terminate when it reaches the end of its code list, even with no (return|exit) statement. A (return|exit) statement is not required to pass a value back from a terminated (function|shell). Every process has a built-in variable $? which always has a numeric value. It is a special variable that cannot be set like "?=1", but it is set only in special ways (see below *). The value of $? after the last command to be executed in the (called function | sub shell) is the value that is passed back to the (function caller | parent shell). That is true whether the last command executed is ("return [n]"| "exit [n]") or plain ("return" or something else which happens to be the last command in the called function's code.

在上面的项目符号列表中,从"(x|y)"中选择,要么总是第一项,要么总是第二项,分别获得关于函数和return的语句,或者shell和exit的语句。

很明显,它们都使用了特殊变量$?在值终止后向上传递值。

*现在是$?可设置:

When a called function terminates and returns to its caller then $? in the caller will be equal to the final value of $? in the terminated function. When a parent shell implicitly or explicitly waits on a single sub shell and is released by termination of that sub shell, then $? in the parent shell will be equal to the final value of $? in the terminated sub shell. Some built-in functions can modify $? depending upon their result. But some don't. Built-in functions "return" and "exit", when followed by a numerical argument both set $? with their argument, and terminate execution.

值得注意的是,$?可以通过在子shell中调用exit来赋值,如下所示:

# (exit 259)
# echo $?
3

在其他一些答案中添加一个可操作的方面:

两者都可以给出退出代码- default或由函数定义,并且唯一的“default”是零,表示成功退出和返回。任何状态都可以有一个自定义数字0-255,包括成功。

返回通常用于运行在当前shell中的交互式脚本,称为with。例如Script.sh,它只是将您返回到调用shell。然后,调用shell可以访问返回代码- $?给出定义的返回状态。 在这种情况下,Exit还会关闭shell(包括SSH连接,如果这是您的工作方式的话)。

如果脚本是可执行的,并且从另一个脚本或shell调用并且在子shell中运行,那么Exit是必要的。然后退出码可以被调用shell访问-在这种情况下return会给出一个错误。

Return将导致当前函数超出作用域,而exit将导致脚本在调用它的地方结束。下面是一个示例程序来帮助解释这一点:

#!/bin/bash

retfunc()
{
    echo "this is retfunc()"
    return 1
}

exitfunc()
{
    echo "this is exitfunc()"
    exit 1
}

retfunc
echo "We are still here"
exitfunc
echo "We will never see this"

输出

$ ./test.sh
this is retfunc()
We are still here
this is exitfunc()

记住,函数是脚本内部的,通常使用return语句返回调用它们的地方。调用外部脚本完全是另一回事,脚本通常以退出语句结束。

“Bash函数中关于退出代码的返回语句和退出语句之间”的差异非常小。两者都返回一个状态,而不是本身的值。状态为0表示成功,而任何其他状态(1到255)表示失败。return语句将从调用它的地方返回到脚本,而exit语句将从遇到它的地方结束整个脚本。

return 0  # Returns to where the function was called.  $? contains 0 (success).

return 1  # Returns to where the function was called.  $? contains 1 (failure).

exit 0  # Exits the script completely.  $? contains 0 (success).

exit 1  # Exits the script completely.  $? contains 1 (failure).

如果函数只是结束而没有返回语句,则最后执行的命令的状态将作为状态代码返回(并将放置在$?)

记住,return和exit返回一个从0到255的状态码,可用$?你不能把任何东西塞进状态码(例如,返回“cat”);这是行不通的。但是,脚本可以通过使用状态码返回255个不同的失败原因。

您可以设置调用脚本中包含的变量,或者在函数中回显结果,并在调用脚本中使用命令替换;但是return和exit的目的是传递状态码,而不是像C这样的编程语言中所期望的值或计算结果。