()、[]、{}、(())、[[]]命令替换

本文详细讲解了Bash脚本中$()和反引号`的命令替换区别,强调了$()的使用优点,并介绍了变量替换、路径操作、整数运算、数组和进制转换等内容,适合初学者和进阶用户参考。

命令替换

在bash中,$( )` `(反引号)都是用来作命令替换的。
命令替换与变量替换差不多,都是用来重组命令行的,先完成引号里的命令行,然后将其结果替换出来,再重组成新的命令行。

exp 1


 
  1. $ echo today is $(date "+%Y-%m-%d")

  2. today is 2014-07-01

 

$( )与``

在操作上,这两者都是达到相应的效果,但是建议使用$( ),理由如下:

  • ``很容易与''搞混乱,尤其对初学者来说。
  • 在多层次的复合替换中,``必须要额外的跳脱处理(反斜线),而$( )比较直观。
  • 最后,$( )的弊端是,并不是所有的类unix系统都支持这种方式,但反引号是肯定支持的。

exp 2


 
  1. # 将cmd1执行结果作为cmd2参数,再将cmd2结果作为cmd3的参数

  2. cmd3 $(cmd2 $(cmd1))

  3.  
  4. # 如果是用反引号,直接引用是不行的,还需要作跳脱处理

  5. cmd3 `cmd2 \`cmd1\``

 

${ }变量替换

一般情况下,$var${var}是没有区别的,但是用${ }会比较精确的界定变量名称的范围


 
  1. $ A=B

  2. $ echo ${A}B

  3. BB

取路径、文件名、后缀
先赋值一个变量为一个路径,如下:
file=/dir1/dir2/dir3/my.file.txt

命令解释结果
${file#*/}拿掉第一条 / 及其左边的字符串dir1/dir2/dir3/my.file.txt
${file##*/}拿掉最后一条 / 及其左边的字符串my.file.txt
${file#*.}拿掉第一个 . 及其左边的字符串file.txt
${file##*.}拿掉最后一个 . 及其左边的字符串txt
${file%/*}拿掉最后一条 / 及其右边的字符串/dir1/dir2/dir3
${file%%/*}拿掉第一条 / 及其右边的字符串(空值)
${file%.*}拿掉最后一个 . 及其右边的字符串/dir1/dir2/dir3/my.file
${file%%.*}拿掉第一个 . 及其右边的字符串/dir1/dir2/dir3/my

记忆方法如下:

  • # 是去掉左边(在键盘上 # 在 $ 之左边)
  • % 是去掉右边(在键盘上 % 在 $ 之右边)
  • 单一符号是最小匹配;两个符号是最大匹配
  • *是用来匹配不要的字符,也就是想要去掉的那部分
  • 还有指定字符分隔号,与*配合,决定取哪部分

取子串及替换

命令解释结果
${file:0:5}提取最左边的 5 个字节/dir1
${file:5:5}提取第 5 个字节右边的连续 5 个字节/dir2
${file/dir/path}将第一个 dir 提换为 path/path1/dir2/dir3/my.file.txt
${file//dir/path}将全部 dir 提换为 path/path1/path2/path3/my.file.txt
${#file}获取变量长度27

根据状态为变量赋值

命令解释备注
${file-my.file.txt}若 $file 没设定,则使用 my.file.txt 作传回值空值及非空值不作处理
${file:-my.file.txt}若 $file 没有设定或为空值,则使用 my.file.txt 作传回值非空值时不作处理
${file+my.file.txt}若$file 设为空值或非空值,均使用my.file.txt作传回值没设定时不作处理
${file:+my.file.txt}若 $file 为非空值,则使用 my.file.txt 作传回值没设定及空值不作处理
${file=txt}若 $file 没设定,则回传 txt ,并将 $file 赋值为 txt空值及非空值不作处理
${file:=txt}若 $file 没设定或空值,则回传 txt ,将 $file 赋值为txt非空值时不作处理
${file?my.file.txt}若 $file 没设定,则将 my.file.txt 输出至 STDERR空值及非空值不作处理
${file:?my.file.txt}若 $file没设定或空值,则将my.file.txt输出至STDERR非空值时不作处理

 

tips:
以上的理解在于, 你一定要分清楚 unset 与 null 及 non-null 这三种赋值状态. 一般而言, : 与 null 有关, 若不带 : 的话, null 不受影响, 若带 : 则连 null 也受影响.

数组


 
  1. A="a b c def" # 定义字符串

  2. A=(a b c def) # 定义字符数组

命令解释结果
${A[@]}返回数组全部元素a b c def
${A[*]}同上a b c def
${A[0]}返回数组第一个元素a
${#A[@]}返回数组元素总个数4
${#A[*]}同上4
${#A[3]}返回第四个元素的长度,即def的长度3
A[3]=xyz则是将第四个组数重新定义为 xyz 

 

$(( ))与整数运算

bash中整数运算符号

符号功能
+ - * /分别为加、减、乘、除
%余数运算
& | ^ !分别为“AND、OR、XOR、NOT”

在 $(( )) 中的变量名称,可于其前面加 $ 符号来替换,也可以不用。


 
  1. $ a=5;b=7;c=2

  2. $ echo $((a+b*c))

  3. 19

  4. $ echo $(($a+$b*$c))

  5. 19

进制转换
$(( ))可以将其他进制转成十进制数显示出来。用法如下:
echo $((N#xx))
其中,N为进制,xx为该进制下某个数值,命令执行后可以得到该进制数转成十进制后的值。


 
  1. $ echo $((2#110)) # 二进制转十进制

  2. 6

  3. $ echo $((16#2a)) # 十六进制转十进制

  4. 42

  5. $ echo $((8#11)) # 八进制转十进制

  6. 9

(( ))重定义变量值


 
  1. $ a=5;b=7

  2. $ ((a++));echo $a

  3. 6

  4. $ ((a--));echo $a

  5. 5

  6.  

$ ((a<b));echo $? 0

使用(( ))作整数测试时,不要跟[ ]的整数测试搞混乱了。

 

在bash中,$( )` `(反引号)都是用来作命令替换的。
命令替换与变量替换差不多,都是用来重组命令行的,先完成引号里的命令行,然后将其结果替换出来,再重组成新的命令行。

exp 1

[root@localhost ~]# echo today is $(date "+%Y-%m-%d")
today is 2017-11-07
[root@localhost ~]# echo today is `date "+%Y-%m-%d"`
today is 2017-11-07

$( )与``
在操作上,这两者都是达到相应的效果,但是建议使用$( ),理由如下:

``很容易与''搞混乱,尤其对初学者来说,而$( )比较直观。
最后,$( )的弊端是,并不是所有的类unix系统都支持这种方式,但反引号是肯定支持的。

exp 2

[root@localhost ~]#  echo Linux `echo Shell `echo today is `date "+%Y-%m-%d"```
Linux Shellecho today is 2017-11-07     #过多使用``会有问题
[root@localhost ~]# echo Linux `echo Shell $(echo today is $(date "+%Y-%m-%d"))`
Linux Shell today is 2017-11-07    ``和$()混合使用
[root@localhost ~]# echo Linux $(echo Shell $(echo today is $(date "+%Y-%m-%d")))
Linux Shell today is 2017-11-07    #多个$()同时使用也不会有问题

${ }变量替换

一般情况下,$var与${var}是没有区别的,但是用${ }会比较精确的界定变量名称的范围

exp 1

[root@localhost ~]# A=Linux
[root@localhost ~]# echo $AB    #表示变量AB

[root@localhost ~]# echo ${A}B    #表示变量A后连接着B
LinuxB

取路径、文件名、后缀

复制代码

先赋值一个变量为一个路径,如下:
file=/dir1/dir2/dir3/my.file.txt

命令    解释    结果
${file#*/}    拿掉第一条 / 及其左边的字符串    dir1/dir2/dir3/my.file.txt
[root@localhost ~]# echo ${file#*/}
dir1/dir2/dir3/my.file.txt

${file##*/}    拿掉最后一条 / 及其左边的字符串    my.file.txt
[root@localhost ~]# echo ${file##*/}
my.file.txt

${file#*.}    拿掉第一个 . 及其左边的字符串    file.txt
[root@localhost ~]# echo ${file#*.}
file.txt

${file##*.}    拿掉最后一个 . 及其左边的字符串    txt
[root@localhost ~]# echo ${file##*.}
txt

${file%/*}    拿掉最后一条 / 及其右边的字符串    /dir1/dir2/dir3
[root@localhost ~]# echo ${file%/*}
/dir1/dir2/dir3

${file%%/*}    拿掉第一条 / 及其右边的字符串    (空值)
[root@localhost ~]# echo ${file%%/*}
(空值)

${file%.*}    拿掉最后一个 . 及其右边的字符串    /dir1/dir2/dir3/my.file
[root@localhost ~]# echo ${file%.*}
/dir1/dir2/dir3/my.file

${file%%.*}    拿掉第一个 . 及其右边的字符串    /dir1/dir2/dir3/my
[root@localhost ~]# echo ${file%%.*}
/dir1/dir2/dir3/my
记忆方法如下:

# 是去掉左边(在键盘上 # 在 $ 之左边)
% 是去掉右边(在键盘上 % 在 $ 之右边)
单一符号是最小匹配;两个符号是最大匹配
*是用来匹配不要的字符,也就是想要去掉的那部分
还有指定字符分隔号,与*配合,决定取哪部分

复制代码

取子串及替换

命令                                    解释                              结果
${file:0:5}               提取最左边的 5 个字节                /dir1
${file:5:5}               提取第 5 个字节右边的连续 5 个字节         /dir2
${file/dir/path}            将第一个 dir 提换为 path              /path1/dir2/dir3/my.file.txt
${file//dir/path}        将全部 dir 提换为 path               /path1/path2/path3/my.file.txt
${#file}              获取变量长度                     27                            

根据状态为变量赋值

命令解释备注
${file-my.file.txt}若 $file 没设定,则使用 my.file.txt 作传回值空值及非空值不作处理
${file:-my.file.txt}若 $file 没有设定或为空值,则使用 my.file.txt 作传回值非空值时不作处理
${file+my.file.txt}若$file 设为空值或非空值,均使用my.file.txt作传回值没设定时不作处理
${file:+my.file.txt}若 $file 为非空值,则使用 my.file.txt 作传回值没设定及空值不作处理
${file=txt}若 $file 没设定,则回传 txt ,并将 $file 赋值为 txt空值及非空值不作处理
${file:=txt}若 $file 没设定或空值,则回传 txt ,将 $file 赋值为txt非空值时不作处理
${file?my.file.txt}若 $file 没设定,则将 my.file.txt 输出至 STDERR空值及非空值不作处理
${file:?my.file.txt}若 $file没设定或空值,则将my.file.txt输出至STDERR非空值时不作处理

 tips:

以上的理解在于, 你一定要分清楚 unset 与 null 及 non-null 这三种赋值状态. 一般而言, : 与 null 有关, 若不带 : 的话, null 不受影响, 若带 : 则连 null 也受影响.

数组

A="a b c def"   # 定义字符串
A=(a b c def)   # 定义字符数组
命令解释结果
${A[@]}返回数组全部元素a b c def
${A[*]}同上a b c def
${A[0]}返回数组第一个元素a
${#A[@]}返回数组元素总个数4
${#A[*]}同上4
${#A[3]}返回第四个元素的长度,即def的长度3
A[3]=xzy则是将第四个组数重新定义为 xyz 

$(( ))与整数运算

 

bash中整数运算符号

符号功能
+ - * /分别为加、减、乘、除
%余数运算
& | ^ !分别为“AND、OR、XOR、NOT”

 在 $(( )) 中的变量名称,可于其前面加 $ 符号来替换,也可以不用。

复制代码

[root@localhost ~]# echo $((2*3))
6
[root@localhost ~]# a=5;b=7;c=2
[root@localhost ~]# echo $((a+b*c))
19
[root@localhost ~]# echo $(($a+$b*$c))
19

复制代码

进制转换

$(( ))可以将其他进制转成十进制数显示出来。用法如下:
echo $((N#xx))
其中,N为进制,xx为该进制下某个数值,命令执行后可以得到该进制数转成十进制后的值。

[root@localhost ~]# echo $((2#110))
6
[root@localhost ~]# echo $((16#2a))
42
[root@localhost ~]# echo $((8#11))
9

(())重定义变量值

复制代码

[root@localhost ~]# a=5;b=7
[root@localhost ~]# ((a++))
[root@localhost ~]# echo $a
6
[root@localhost ~]# ((a--));echo $a
5
[root@localhost ~]# ((a<b));echo $?
0
[root@localhost ~]# ((a>b));echo $?
1
### 命令替换使用方法 命令替换允许将一个命令的输出作为另一个命令的参数。在 shell 中有两种常见的语法来实现命令替换: - 反引号(`` ` ``):例如,要将 `ls` 命令的输出作为参数传递给 `wc -l` 命令以统计文件数量,可以使用 ``wc -l `ls` ``。 - `$()` 语法:这是更现代的语法,例如 `wc -l $(ls)`。它的优点是可以嵌套使用,如 `echo $(date +%Y-%m-%d) $(ls -l | wc -l)`。 ### xargs 使用方法 `xargs` 用于将标准输入转换为命令行参数。大多数命令与管道一起使用时,需要 `xargs` 将标准输入转为命令行参数。例如,`echo "one two three" | xargs mkdir` 会根据标准输入创建名为 `one``two``three` 的目录。`xargs` 还有很多选项,如 `-0``\0` 作为输入分隔符,可处理包含特殊字符的文件名;`-n` 指定每次传递给命令的参数数量,`-L` 指定每次从输入读取的行数等 [^1]。 ### 命令替换和 xargs 的区别 - **数据来源**: - 命令替换是将一个命令的输出直接嵌入到另一个命令的参数中,它是在命令执行前完成替换的。 - `xargs` 是从标准输入读取数据,并将其作为参数传递给指定的命令,通常与管道结合使用。 - **处理方式**: - 命令替换会将整个命令输出作为一个整体进行处理,如果输出包含空格等分隔符,可能会导致参数分割不符合预期。 - `xargs` 可以根据指定的分隔符对输入进行分割,并逐一组装成命令行参数,能更好地处理包含特殊字符的输入。 - **性能方面**:当处理大量数据时,`xargs` 通常更高效,因为它可以批量处理参数,而命令替换可能会因为参数过长导致问题。 ### 结合使用场景 - **批量文件处理**:例如,要对 `find` 命令查找出的文件进行批量操作。可以使用命令替换先获取文件列表,再结合 `xargs` 进行处理。 ```bash find /path/to/search -type f -name "*.txt" -print0 | xargs -0 tar -cvf $(date +%Y%m%d).tar ``` 这里先使用 `find` 查找所有 `.txt` 文件,通过 `xargs` 将这些文件名传递给 `tar` 命令,同时使用命令替换获取当前日期作为压缩包的文件名。 - **动态生成命令参数**:当需要根据某个命令的输出动态生成参数,并将其传递给另一个命令时,可以结合使用。例如,根据 `grep` 查找的结果,使用 `xargs` 执行相应的操作。 ```bash grep -rl "pattern" /path/to/search | xargs sed -i 's/pattern/replacement/g' ``` 这里 `grep -rl` 查找包含指定模式的文件,将结果通过管道传递给 `xargs`,再由 `xargs` 将这些文件名传递给 `sed` 命令进行替换操作。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

生信小博士

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值