Bash 初学者系列 9:在 bash 中使用函数

系统 Linux
今天我们将介绍在 bash 中如何创建函数,以及函数中的返回值、传递参数等内容。

当你的 bash 脚本中代码非常多的时候,会看起来很混乱,其中有部分代码有可能是重复的。这个时候,可以使用函数来避免重复的代码。

今天我们将介绍在 bash 中如何创建函数,以及函数中的返回值、传递参数等内容。

在 bash 中创建函数

创建 bash 函数有两种不同的语法。最常用的方法如下:​

function_name () {
commands
}

第二种方法不太常用,如下所示:

function function_name {
commands
}

在使用函数时,需要注意以下几点:

  • 除非被调用,否则函数永远不会被执行;
  • 函数必须先定义,然后才能被调用。

作为演示,我们创建一个 fun.sh 脚本,如下代码:

#!/bin/bash

hello () {
echo "Hello World"
}

hello
hello
hello

上述脚本中,我们定义了一个名为 hello 的函数,它将在终端上打印 Hello World。后面我们调用了三次 hello 函数,因此,运行脚本后,你会在屏幕上看到 Hello World 打印了三次:

$ ./fun.sh
Hello World
Hello World
Hello World

函数的返回值

在很多编程语言中,函数在调用时都会返回一个值,然而,在 bash 函数中没有返回值。

当函数执行完成时,会返回一个 $? 变量作为退出状态,0 表示成功执行,其他非零正整数(1 - 255)表示执行失败。

我们来写一个例子,命名为 error.sh,在其中写一个 return 语句,如下代码所示:​

#! /bin/bash

error () {
blabla
return 0
}

error
echo "The return status of the error function is: $?"

执行结果如下:

$ ./error.sh
./error.sh: line 4: blabla: command not found
The return status of the error function is: 0

因为有 return 0 这一行,所以即使函数中有 blabla 这一行的报错,error 函数还是返回了一个执行成功的状态码 0。

return 语句执行后,函数会被立即终止。

向 bash 函数传递参数

我们可以向函数传递参数,就像向 bash 脚本传递参数​一样。

作为演示,我们创建一个名为 iseven.sh 的脚本,如下所示:​

#!/bin/bash

iseven () {
if [ $(($1 % 2)) -eq 0 ]; then
echo "$1 is even."
else
echo "$1 is odd."
fi
}

iseven 3
iseven 4
iseven 20
iseven 111

iseven() 函数判断数字是偶数还是奇数。在调用的时候,将参数放在函数名后面,函数执行的时候会读取 $1 来获取传递的参数。我们来执行一下:

$ ./iseven.sh
3 is odd.
4 is even.
20 is even.
111 is odd.

这里需要注意,bash 函数中的参数,和 bash 脚本的参数需要区分开来,看如下 funarg.sh 脚本:​

#!/bin/bash

fun () {
echo "$1 is the first argument to fun()"
echo "$2 is the second argument to fun()"
}

echo "$1 is the first argument to the script."
echo "$2 is the second argument to the script."

fun Yes 7

执行结果如下:

$ ./funarg.sh Cool Stuff
Cool is the first argument to the script.
Stuff is the second argument to the script.
Yes is the first argument to fun()7 is the second argument to fun()

可以看到,同样是使用 $1 $2 来引用脚本参数和函数参数,当实际调用时,取值是不同的。

bash 函数中的全局变量和局部变量

与大多数编程语言累死,bash 变量具有全局变量和局部变量之分。其中全局变量可以在任何位置方位到,而局部变量只能在其定义的函数中访问。

作为演示,看如下 scope.sh 脚本:​

#!/bin/bash

v1='A'
v2='B'

myfun() {
local v1='C'
v2='D'
echo "Inside myfun(): v1: $v1, v2: $v2"
}

echo "Before calling myfun(): v1: $v1, v2: $v2"
myfun
echo "After calling myfun(): v1: $v1, v2: $v2"

我们首先定义了两个全局变量 v1 v2,然后在 函数 myfun() 中,使用 local 关键字 定义了一个局部变量 v1,并修改了全局变量 v2。在不同的函数中,局部变量可以使用相同的变量名。

执行一下,结果如下:​

$ ./scope.sh
Before calling myfun(): v1: A, v2: B
Inside myfun(): v1: C, v2: D
After calling myfun(): v1: A, v2: D


由此,我们可以看到:

  • 如果局部变量的变量名与全局变量的变量名相同,那么局部变量优先级会更高;
  • 在函数中可以更改全局变量的值。

递归函数

递归函数是一个调用自身的函数。阶乘计算是递归函数的经典例子,看下面的脚本 factorial.sh:

#!/bin/bash

factorial () {
if [ $1 -le 1 ]; then
echo 1
else
last=$(factorial $(( $1 -1)))
echo $(( $1 * last ))
fi
}

echo -n "4! is: "
factorial 4
echo -n "5! is: "
factorial 5
echo -n "6! is: "
factorial 6

任何递归函数都要从一个基本条件开始,这个基本条件必须可以结束递归函数调用链。在 factorial() 函数中,基本条件为:

if [ $1 -le 1 ]; then
echo 1

然后导出阶乘函数的递归情况。要计算 n 的阶乘,其中 n 是大于 1 的正数,可以将 n 乘以 n-1 的阶乘:

factorial(n) = n * factorial(n-1)

使用上面那个公式来编写递归函数的算法:​

last=$(factorial $(( $1 -1)))
echo $(( $1 * last ))

然后运行一下,检查运行结果:

$ ./factorial.sh
4! is: 24
5! is: 120
6! is: 720


责任编辑:庞桂玉 来源: TIAP
相关推荐

2022-12-02 14:30:24

Bash脚本数组

2022-12-13 12:55:15

Bash循环

2022-12-16 09:01:24

2022-12-01 08:10:49

Bash脚本参数

2022-12-07 07:43:21

2022-12-09 07:58:52

Bash条件语句

2022-12-08 12:05:03

Bash字符串

2022-11-30 07:47:00

Bash脚本

2022-11-28 08:01:47

BashLinuxshell 脚本

2023-07-12 14:13:03

BashLinux

2023-07-20 15:37:50

Bash数组

2023-08-22 21:42:03

Bash函数

2020-10-13 19:04:58

Bash信号捕获Shell脚本

2019-04-22 14:39:05

BashLinux命令

2019-04-15 11:10:34

Bash方括号命令行

2011-06-17 15:37:42

Qt

2022-01-16 07:48:33

脚本Bash 工具

2023-01-13 12:37:43

Bashshell花括号

2012-03-14 10:56:23

web app

2020-04-09 10:18:51

Bash循环Linux
点赞
收藏

51CTO技术栈公众号