shell中方括号在比较中的使用规则


原文链接: shell中方括号在比较中的使用规则

(括号)的作用

Shell中的括号有其特殊的用法, 现总结如下:

  1. 符号$后的括号
    ${a} 变量a的值, 在不引起歧义的情况下可以省略大括号.
    $(cmd) 命令替换, 结果为shell命令cmd的输出, 和cmd效果相同, 不过某些Shell版本不支持$()形式的命令替换, 如tcsh.
    $((exp)) 和expr exp效果相同, 计算数学表达式exp的数值, 其中exp只要符合C语言的运算规则即可, 甚至三目运算符和逻辑表达式都可以计算.

  2. 多条命令执行
    (cmd1;cmd2;cmd3) 新开一个子shell顺序执行命令cmd1,cmd2,cmd3, 各命令之间用分号隔开, 最后一个命令后可以没有分号.
    { cmd1;cmd2;cmd3;} 在当前shell顺序执行命令cmd1,cmd2,cmd3, 各命令之间用分号隔开, 最后一个命令后必须有分号, 第一条命令和左括号之间必须用空格隔开.
    对{}和()而言, 括号中的重定向符只影响该条命令, 而括号外的重定向符影响到括号中的所有命令.

  3. 双括号的特殊用法
    (()) 增强括号的用法, 常用于算术运算比较. 双括号中的变量可以不使用$符号前缀, 只要括号中的表达式符合c语言运算规则, 支持多个表达式用逗号分开.
    比如可以直接使用for((i=0;i<5;i++)), 如果不使用双括号, 则为for i in `seq 0 4`或者for i in {0..4}. 再如可以直接使用if (($i<5)), 如果不使用双括号, 则为if [ $i -lt 5 ]. [[]] 增强方括号用法, 常用于字符串的比较. 主要用于条件测试, 双括号中的表达式可以使用&&, ||, <, >等C语言语法.
    比如可以直接使用if [[ $a != 1 && $a != 2 ]], 如果不适用双括号, 则为if [ $a -ne 1] && [ $a != 2 ]或者if [ $a -ne 1 -a $a != 2 ].

[方括号]在比较中的使用规则

一. bash [ ] 单方括号的使用

基本要素:

  1. 内部操作符与操作变量之间要有空格:如 [ "a" = "b" ]
  2. 字符串比较中,> < 需要写成> < 进行转义

举例: [ -w result.txt-a -w score.txt ] ;echo $? // 测试两个文件是否均可写

二. bash [[ ]] 双方括号的使用
基本要素:

  1. [[ ]] 两个符号左右都要有空格分隔
  2. 内部操作符与操作变量之间要有空格:如 [[ "a" = "b" ]]
  3. 字符串比较中,可以直接使用 > < 无需转义
  4. [[ ]] 中字符串或者${}变量尽量如未使用"" 双引号扩住的话,会进行模式和元字符匹配
    [[ "ab"=a* ]] && echo "ok"
  5. [[ ]] 内部可以使用 && || 进行逻辑运算
  6. [[ ]] 是bash keyword:[[ is a shell keyword
    [[ ]] 其他用法都和[ ] 一样

二者共同特性:
&& ||-a -o 处理
[ exp1 -a exp2 ] = [[ exp1 && exp2 ]] = [ exp1 ]&& [ exp2 ] = [[ exp1 ]] && [[ exp2 ]]
[ exp1 -o exp2 ] = [[ exp1 || exp2 ]] = [ exp1 ]|| [ exp2 ] = [[ exp1 ]] || [[ exp2 ]]

[root@localhost ~]# if [[ "a" == "a" && 2 -gt1 ]] ;then echo "ok" ;fi
[root@localhost ~]# if [[ "a" == "a" ]] && [[2 -gt 1 ]] ;then echo "ok" ;fi
[root@localhost ~]# if [[ "a" == "a" ]] || [[ 2 -gt 1]] ;then echo "ok" ;fi
[root@localhost ~]# if [[ "a" == "a" ]] || [[ 2 -gt10 ]] ;then echo "ok" ;fi
[root@localhost ~]# if [[ "a" == "a" || 2 -gt 10 ]] ;then echo "ok" ;fi
[[ ]] 和 [ ] 都可以和 ! 配合使用

优先级 ! > && > ||
逻辑运算符 < 关系运算符
逻辑运算符 : ! && || -a -o
关系运算符 : < > > < == = != - eq -ne -gt -ge -lt -le

[[ ]] 比[ ] 具备的优势


①[[是 bash 程序语言的关键字。并不是一个命令,[[ ]] 结构比[ ]结构更加通用。在[[和]]之间所有的字符都不会发生文件名扩展或者单词分割,但是会发生参数扩展和命令替换。
②支持字符串的模式匹配,使用=~操作符时甚至支持shell的正则表达式。字符串比较时可以把右边的作为一个模式,而不仅仅是一个字符串,比如[[ hello == hell? ]],结果为真。[[ ]] 中匹配字符串或通配符,不需要引号。
③使用[[ ... ]]条件判断结构,而不是[... ],能够防止脚本中的许多逻辑错误。比如,&&、||、<和> 操作符能够正常存在于[[ ]]条件判断结构中,但是如果出现在[ ]结构中的话,会报错。
④bash把双中括号中的表达式看作一个单独的元素,并返回一个退出状态码。

使用[[ ... ]]条件判断结构, 而不是[ ... ], 能够防止脚本中的许多逻辑错误. 比如,&&, ||, <, 和> 操作符能够正常存在于[[]]条件判断结构中, 但是如果出现在[ ]结构中的话, 会报错。

if 条件判断中有多个条件
#!/bin/bash
score=$1
if [ $score = 5 ]||[ $score = 3 ];then

echo right

else

echo wrong

fi


#!/bin/bash
score=$1
if [ $score -gt 5 ]||[ $score -lt 3 ];then

echo right

else

echo wrong

fi


#!/bin/bash
score=$1
if [ $score -gt 15 ]||([ $score -lt 8 ]&&[ $score -ne 5 ]);then

echo right

else

echo wrong

fi


或:
#!/bin/bash

count="$1"

if [ $count -gt 15 -o $count -lt 5 ];then

echo right

fi

且:
#!/bin/bash

count="$1"

if [ $count -gt 5 -a $count -lt 15 ];then

echo right

fi

score=$1
if [[ $score -gt 15 || $score -lt 8 && $score -ne 5 ]];then

echo right

else

echo wrong

fi
记住必须加两个中括号

`