`
icyheart
  • 浏览: 775909 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

Bash的测试和比较函数

阅读更多

您是否为 Bash shell 中大量的测试和比较选项而困惑呢?这个技巧可以帮助您解密不同类型的文件、算术和字符串测试,这样您就能够知道什么时候使用 test、 [ ]、 [[ ]]、 (( )) 或 if-then-else 了。

Bash shell 在当今的许多 Linux® 和 UNIX® 系统上都可使用,是 Linux 上常见的默认 shell。Bash 包含强大的编程功能,其中包括丰富的可测试文件类型和属性的函数,以及在多数编程语言中可以使用的算术和字符串比较函数。理解不同的测试并认识到 shell 还能把一些操作符解释成 shell 元字符,是成为高级 shell 用户的重要一步。这篇文章摘自 developerWorks 教程 LPI exam 102 prep: Shells, scripting, programming, and compiling,介绍了如何理解和使用 Bash shell 的测试和比较操作。


这个技巧解释了 shell 测试和比较函数,演示了如何向 shell 添加编程功能。您可能已经看到过使用 && 和 || 操作符的简单 shell 逻辑,它允许您根据前一条命令的退出状态(正确退出或伴随错误退出)而执行后一条命令。在这个技巧中,将看到如何把这些基本的技术扩展成更复杂的 shell 编程。


测试


在任何一种编程语言中,学习了如何给变量分配值和传递参数之后,都需要测试这些值和参数。在 shell 中,测试会设置返回的状态,这与其他命令执行的功能相同。实际上,test 是个内置命令!


test 和 [


内 置命令 test 根据表达式expr 求值的结果返回 0(真)或 1(假)。也可以使用方括号:test expr 和 [ expr ] 是等价的。 可以用 $? 检查返回值;可以使用 && 和 || 操作返回值;也可以用本技巧后面介绍的各种条件结构测试返回值。


清单 1. 一些简单测试

[ian@pinguino ~]$ test 3 -gt 4 && echo True || echo false

false

[ian@pinguino ~]$ [ "abc" != "def" ];echo $?

0

[ian@pinguino ~]$ test -d "$HOME" ;echo $?

0


在清单 1 的第一个示例中,-gt 操作符对两个字符值之间执行算术比较。在第二个示例中,用 [ ] 的形式比较两个字符串不相等。在最后一个示例中,测试 HOME 变量的值,用单目操作符 -d 检查它是不是目录。


可以用 -eq、 -ne、-lt、 -le、 -gt 或 -ge 比较算术值,它们分别表示等于、不等于、小于、小于等于、大于、大于等于。


可以分别用操作符 =、 !=、< 和 > 比较字符串是否相等、不相等或者第一个字符串的排序在第二个字符串的前面或后面。单目操作符 -z 测试 null 字符串,如果字符串非空 -n 返回 True(或者根本没有操作符)。


说明:shell 也用 < 和 > 操作符进行重定向,所以必须用 \< 或 \> 加以转义。清单 2 显示了字符串测试的更多示例。检查它们是否如您预期的一样。


清单 2. 一些字符串测试

[ian@pinguino ~]$ test "abc" = "def" ;echo $?

1

[ian@pinguino ~]$ [ "abc" != "def" ];echo $?

0

[ian@pinguino ~]$ [ "abc" \< "def" ];echo $?

0

[ian@pinguino ~]$ [ "abc" \> "def" ];echo $?

1

[ian@pinguino ~]$ [ "abc" \<"abc" ];echo $?

1

[ian@pinguino ~]$ [ "abc" \> "abc" ];echo $?

1


表 1 显示了一些更常见的文件测试。如果被测试的文件存在,而且有指定的特征,则结果为 True。


表 1. 一些常见的文件测试 操作符 特征

-d 目录

-e 存在(也可以用 -a)

-f 普通文件

-h 符号连接(也可以用 -L)

-p 命名管道

-r 可读

-s 非空

-S 套接字

-w 可写

-N 从上次读取之后已经做过修改


除了上面的单目测试,还可以使用表 2 所示的双目操作符比较两个文件:


表 2. 测试一对文件 操作符 为 True 的情况

-nt 测试 file1 是否比 file2 更新。修改日期将用于这次和下次比较。

-ot 测试 file1 是否比 file2 旧。

-ef 测试 file1 是不是 file2 的硬链接。


其他一些测试可以用来测试文件许可之类的内容。请参阅 bash 手册获得更多细节或使用 help test 查看内置测试的简要信息。也可以用 help 命令了解其他内置命令。


-o 操作符允许测试利用 set -o 选项 设置的各种 shell 选项,如果设置了该选项,则返回 True (0),否则返回 False (1),如清单 3 所示。


清单 3. 测试 shell 选项

[ian@pinguino ~]$ set +o nounset

[ian@pinguino ~]$ [ -o nounset ];echo $?

1

[ian@pinguino ~]$ set -u

[ian@pinguino ~]$ test -o nounset; echo $?

0


最 后,-a 和 -o 选项允许使用逻辑运算符 AND 和 OR 将表达式组合在一起。单目操作符 ! 可以使测试的意义相反。可以用括号把表达式分组,覆盖默认的优先级。请记住 shell 通常要在子 shell 中运行括号中的表达式,所以需要用 \( 和 \) 转义括号,或者把这些操作符括在单引号或双引号内。清单 4 演示了摩根法则在表达式上的应用。


清单 4. 组合和分组测试

[ian@pinguino ~]$ test "a" != "$HOME" -a 3 -ge 4 ; echo $?

1

[ian@pinguino ~]$ [ ! \( "a" = "$HOME" -o 3 -lt 4 \) ]; echo $?

1

[ian@pinguino ~]$ [ ! \( "a" = "$HOME" -o '(' 3 -lt 4 ')' ")" ]; echo $?

1


(( 和 [[


test 命令非常强大,但是很难满足其转义需求以及字符串和算术比较之间的区别。幸运的是,bash 提供了其他两种测试方式,这两种方式对熟悉 C、C++ 或 Java® 语法的人来说会更自然些。


(( )) 复合命令 计算算术表达式,如果表达式求值为 0,则设置退出状态为 1;如果求值为非 0 值,则设置为 0。不需要对 (( 和 )) 之间的操作符转义。算术只对整数进行。除 0 会产生错误,但不会产生溢出。可以执行 C 语言中常见的算术、逻辑和位操作。 let 命令也能执行一个或多个算术表达式。它通常用来为算术变量分配值。


清单 5. 分配和测试算术表达式

[ian@pinguino ~]$ let x=2 y=2**3 z=y*3;echo $? $x $y $z

0 2 8 24

[ian@pinguino ~]$ (( w=(y/x) + ( (~ ++x) & 0x0f ) )); echo $? $x $y $w

0 3 8 16

[ian@pinguino ~]$ (( w=(y/x) + ( (~ ++x) & 0x0f ) )); echo $? $x $y $w

0 4 8 13


同使用 (( )) 一样,利用复合命令 [[ ]] 可以对文件名和字符串使用更自然的语法。可以用括号和逻辑操作符把 test 命令支持的测试组合起来。


清单 6. 使用 [[ 复合命令

[ian@pinguino ~]$ [[ ( -d "$HOME" ) && ( -w "$HOME" ) ]] &&

> echo "home is a writable directory"

home is a writable directory


在使用 = 或 != 操作符时,复合命令 [[ 还能在字符串上进行模式匹配。匹配的方式就像清单 7 所示的通配符匹配。


清单 7. 用 [[ 进行通配符测试

[ian@pinguino ~]$ [[ "abc def .d,x--" == a[abc]*\ ?d* ]]; echo $?

0

[ian@pinguino ~]$ [[ "abc def c" == a[abc]*\ ?d* ]]; echo $?

1

[ian@pinguino ~]$ [[ "abc def d,x" == a[abc]*\ ?d* ]]; echo $?

1


甚至还可以在 [[ 复合命令内执行算术测试,但是千万要小心。除非在 (( 复合命令内,否则 < 和 > 操作符会把操作数当成字符串比较并在当前排序序列中测试它们的顺序。清单 8 用一些示例演示了这一点。


清单 8. 用 [[ 包含算术测试

[ian@pinguino ~]$ [[ "abc def d,x" == a[abc]*\ ?d* || (( 3 > 2 )) ]]; echo $?

0

[ian@pinguino ~]$ [[ "abc def d,x" == a[abc]*\ ?d* || 3 -gt 2 ]]; echo $?

0

[ian@pinguino ~]$ [[ "abc def d,x" == a[abc]*\ ?d* || 3 > 2 ]]; echo $?

0

[ian@pinguino ~]$ [[ "abc def d,x" == a[abc]*\ ?d* || a > 2 ]]; echo $?

0

[ian@pinguino ~]$ [[ "abc def d,x" == a[abc]*\ ?d* || a -gt 2 ]]; echo $?

-bash: a: unbound variable

条件测试


虽然使用以上的测试和 &&、 || 控制操作符能实现许多编程,但 bash 还包含了更熟悉的 “if, then, else” 和 case 结构。学习完这些之后,将学习循环结构,这样您的工具箱将真正得到扩展。


If、then、else 语句


bash 的 if 命令是个复合命令,它测试一个测试或命令($?)的返回值,并根据返回值为 True(0)或 False(不为 0)进行分支。虽然上面的测试只返回 0 或 1 值,但命令可能返回其他值。请参阅 LPI exam 102 prep: Shells, scripting, programming, and compiling 教程学习这方面的更多内容。


Bash 中的 if 命令有一个 then 子句,子句中包含测试或命令返回 0 时要执行的命令列表,可以有一个或多个可选的 elif 子句,每个子句可执行附加的测试和一个 then 子句,子句中又带有相关的命令列表,最后是可选的 else 子句及命令列表,在前面的测试或 elif 子句中的所有测试都不为真的时候执行,最后使用 fi 标记表示该结构结束。


使用迄今为止学到的东西,现在能够构建简单的计算器来计算算术表达式,如清单 9 所示:


清单 9. 用 if、then、else 计算表达式

[ian@pinguino ~]$ function mycalc ()

> {

> local x

> if [ $# -lt 1 ]; then

> echo "This function evaluates arithmetic for you if you give it some"

> elif (( $* )); then

> let x="$*"

> echo "$* = $x"

> else

> echo "$* = 0 or is not an arithmetic expression"

> fi

> }

[ian@pinguino ~]$ mycalc 3 + 4

3 + 4 = 7

[ian@pinguino ~]$ mycalc 3 + 4**3

3 + 4**3 = 67

[ian@pinguino ~]$ mycalc 3 + (4**3 /2)

-bash: syntax error near unexpected token `('

[ian@pinguino ~]$ mycalc 3 + "(4**3 /2)"

3 + (4**3 /2) = 35

[ian@pinguino ~]$ mycalc xyz

xyz = 0 or is not an arithmetic expression

[ian@pinguino ~]$ mycalc xyz + 3 + "(4**3 /2)" + abc

xyz + 3 + (4**3 /2) + abc = 35


这个计算器利用 local 语句将 x 声明为局部变量,只能在 mycalc 函数的范围内使用。let 函数具有几个可用的选项,可以执行与它密切关联的 declare 函数。请参考 bash 手册或使用 help let 获得更多信息。


如清单 9 所示,需要确保在表达式使用 shell 元字符 —— 例如(、)、*、> 和 < 时 —— 正确地对表达式转义。无论如何,现在有了一个非常方便的小计算器,可以像 shell 那样进行算术计算。


在清单 9 中可能注意到 else 子句和最后的两个示例。可以看到,把 xyz 传递给 mycalc 并没有错误,但计算结果为 0。这个函数还不够灵巧,不能区分最后使用的示例中的字符值,所以不能警告用户。可以使用字符串模式匹配测试(例如

[[ ! ("$*" == *[a-zA-Z]* ]]

, 或使用适合自己范围的形式)消除包含字母表字符的表达式,但是这会妨碍在输入中使用 16 进制标记,因为使用 16 进制标记时可能要用 0x0f 表示 15。实际上,shell 允许的基数最高为 64(使用 base#value 标记),所以可以在输入中加入 _ 和 @ 合法地使用任何字母表字符。8 进制和 16 进制使用常用的标记方式,开头为 0 表示八进制,开头为 0x 或 0X 表示 16 进制。清单 10 显示了一些示例。


清单 10. 用不同的基数进行计算

[ian@pinguino ~]$ mycalc 015

015 = 13

[ian@pinguino ~]$ mycalc 0xff

0xff = 255

[ian@pinguino ~]$ mycalc 29#37

29#37 = 94

[ian@pinguino ~]$ mycalc 64#1az

64#1az = 4771

[ian@pinguino ~]$ mycalc 64#1azA

64#1azA = 305380

[ian@pinguino ~]$ mycalc 64#1azA_@

64#1azA_@ = 1250840574

[ian@pinguino ~]$ mycalc 64#1az*64**3 + 64#A_@

64#1az*64**3 + 64#A_@ = 1250840574


对输入进行的额外处理超出了本技巧的范围,所以请小心使用这个计算器。


elif 语句非常方便。它允许简化缩进,从而有助于脚本编写。在清单 11 中可能会对 type 命令在 mycalc 函数中的输出感到惊讶。


清单 11. Type mycalc

[ian@pinguino ~]$ type mycalc

mycalc is a function

mycalc ()

{

local x;

if [ $# -lt 1 ]; then

echo "This function evaluates arithmetic for you if you give it some";

else

if (( $* )); then

let x="$*";

echo "$* = $x";

else

echo "$* = 0 or is not an arithmetic expression";

fi;

fi

}


当 然,也可以只用 $(( 表达式 )) 和 echo 命令进行 shell 算术运算,如清单 12 所示。这样就不必学习关于函数或测试的任何内容,但是请注意 shell 不会解释元字符,例如 *,因此元字符不能在 (( 表达式 )) 或 [[ 表达式 ]] 中那样正常发挥作用。


清单 12. 在 shell 中用 echo 和 $(( )) 直接进行计算

[ian@pinguino ~]$ echo $((3 + (4**3 /2)))

0
0
分享到:
评论

相关推荐

    Linux使用技巧:Bash测试和比较函数

    总之,理解和熟练运用Bash中的测试和比较函数是提升Linux使用技能的关键。无论是简单的文件检查,还是复杂的条件逻辑,Bash都提供了强大的工具来满足各种需求。通过学习这些技巧,您可以更有效地自动化任务,编写出...

    linux-bashassert一个轻量级的bash单元测试框架

    3. **真值断言**:`assert_true`和`assert_false`分别用来测试一个表达式或命令的返回值是否为真(非零)或假(零),这对于测试条件语句和函数返回值很有帮助。 4. **包含断言**:`assert_in_list`检查一个值是否...

    高级bash shell手册

    `.bashrc` 和 `.bash_profile` 是Bash的启动配置文件,用来设置环境变量、别名和函数,这些设置只影响当前用户。 **13. Shell脚本最佳实践** 编写高效、可维护的Bash脚本需要遵循一定的最佳实践,如注释、错误处理...

    bash高级编程 advanced bash script

    3. **条件测试与文件操作**:掌握测试表达式,用于检查文件类型、文件存在性、数值比较等,并学习如何进行文件的创建、移动、复制和删除。 4. **数组与关联数组**:Bash从4.0版本开始支持关联数组,这允许我们存储...

    bash入门学习实例

    可以测试文件属性、数值比较、字符串操作等。 ### 6. 文件名扩展和通配符 - `*`: 匹配任意字符序列(不包括路径分隔符)。 - `?`: 匹配任意单个字符。 - `[char1-char2]`: 匹配指定范围内的任何字符。 ### 7. ...

    详细的Bash介绍

    使用`if`语句进行条件测试,可以测试文件属性、数值比较、字符串比较等。 7. **流程控制** Bash支持`for`、`while`循环以及`case`、`if`、`else`等控制结构。 8. **函数** Bash允许定义函数,通过`function`...

    bash-fun: bash中的函数式编程

    在Bash中,编写纯函数意味着函数不修改全局变量,只依赖于其参数,这样可以提高代码的可预测性和可测试性。 8. **闭包**: Bash不直接支持闭包,但可以通过函数嵌套和局部变量模拟闭包的概念。当一个内部函数能...

    bash脚本基础实例

    以上五个实例涵盖了Bash脚本的基本元素:变量、运算、条件控制、循环和函数。通过理解和实践这些例子,你可以开始编写自己的Bash脚本,实现更多复杂的自动化任务。记住,Bash的强大之处在于其灵活性和与其他Linux...

    高级Bash脚本编程指南(Advanced Bash-Scripting Guide)中英文版本

    **ABS(Advanced Bash-Scripting Guide)指南详细介绍了Bash shell的各种特性,包括变量、控制结构、函数、输入/输出重定向、条件测试、正则表达式等核心概念。** 1. **变量**:在Bash脚本中,变量用于存储数据,...

    BASH中文版教程

    5. **函数和脚本调试**:创建自定义命令,以及使用set命令和trap命令进行脚本调试。 6. **进程控制**:子shell、后台执行、信号处理和进程间通信。 7. **环境变量和配置文件**:了解如何设置和使用影响shell行为的...

    高级Bash 脚本编程指南 PDF

    Bash支持在字符串比较中使用正则表达式,这在文件名匹配、文本搜索等方面非常实用。 ### 键盘绑定和自定义 shell 通过`.bashrc`文件,可以自定义Bash的键盘快捷键和环境设置。这使得Bash更加符合个人工作习惯。 #...

    Bash4.0参考文档.zip

    Bash提供了丰富的条件测试结构,如`[ ]`和`[[ ]]`用于逻辑判断,`if...then...else...fi`、`case...esac`用于复杂条件分支。同时,`for`、`while`和`until`循环结构帮助实现重复执行。 ### 4. 函数 Bash支持用户...

    Bash中关于日期时间操作的常用自定义函数

    在Bash shell脚本编程中,处理日期和时间是一项常见的任务。这篇博文主要探讨了如何在Bash中创建自定义函数来实现灵活的日期时间操作。以下是一些关键知识点: 1. **`date` 命令**: `date` 是Linux系统中的一个...

    bash-3.1.tar.gz

    5. **壳程序编程**:增加了对shell脚本编写的支持,包括条件测试、循环结构和函数等。 通过这个源代码,开发者或系统管理员可以对Bash进行深度定制,以适应特定的需求或环境,同时也可以查看和学习其内部工作原理。...

    高级Bash脚本编程指南 操作系统 - Linux - 高级Bash脚本编程指南.zip

    标签中提到了C#,虽然两者属于不同领域,但理解Bash脚本对于C#开发者同样重要,因为Bash常常用于自动化构建、部署和测试过程,特别是在Linux服务器上。 总的来说,“高级Bash脚本编程指南”将带你深入探索Bash的...

    最经典的bash资料

    3. **条件测试和逻辑运算**:理解`if`、`else`、`case`语句,以及如何使用条件测试表达式(如`-f`、`-d`等)进行文件类型检查。 4. **流程控制**:掌握`for`、`while`、`until`循环,以及`break`、`continue`语句的...

    linux高级bash编程(abs)

    ]]`或`test`命令进行字符串和模式的比较。 **10. 管道和进程间通信** Bash脚本可以通过管道、重定向和套接字进行进程间通信,实现更复杂的任务协调。 总之,高级Bash Shell编程涵盖了广泛的概念和技术,从基本的...

    Pure Bash Bible 中文版 PDF

    - **文件比较**,**文件是否存在**,**创建空文件**:Bash的测试命令如`test`或`[ ]`可以进行文件操作和条件判断。 9. **控制流**: - **三元运算符**,**陷阱(trap)**,**忽略终端中断**,**窗口大小变化响应*...

    linux bash shell中文手册

    17. **条件表达式 (CONDITIONAL EXPRESSIONS)**:用于进行逻辑判断的表达式,例如比较操作,文件测试等。 18. **简单命令扩展 (SIMPLE COMMAND EXPANSION)**:描述如何解析和扩展简单的命令行输入,包括查找命令、...

    shell bash 手册

    函数和子Shell 可以在Bash脚本中定义函数,并且函数可以调用其他函数或者创建新的子Shell环境。 #### 4. 特殊变量 Bash提供了一系列特殊变量,如`$?`用于获取上一个命令的退出状态。 #### 5. 文件描述符 可以使用...

Global site tag (gtag.js) - Google Analytics