Shell中命令替换$(...)与后置引用`...`的区别与联系

一. 命令替换

a.在bash中,$( )与``(反引号)都是用来作命令替换的。

b.命令替换与变量替换差不多,都是用来重组命令行的,先完成引号里的命令行,然后将其结果替换出来,再重组成新的命令行。

例子1:

$ echo today is $(date "+%Y-%m-%d")
today is 2014-07-01

二. $( )与``(反引号)

(1)区别

    1. `CMD`在执行时,shell会不管``内是什么都先进行解释,再把解释后的最终结果送给shell去执行.如果解释后的最终结果不是shell可执行的命令时,则会出错.当然,仅仅为了把CMD执行后的内容作为文本输出,则没什么问题了啦.
    2. $(CMD)在执行时,如果CMD是命令,则直接丢给shell去执行;如果是变量取值,则也仅作第一层的字面解释后丢给shell去执行.

(2)使用场合

    1. ``用在产生的结果不会再送给shell解释,而只作为赋值时直接使用,文本输出时与""配合使用;
    2. $()有在产生的结果还会再作进一步解释时,用与不用都可以,只是提高程序可读性之外。

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

    1. ``很容易与‘‘搞混乱。
    2. 在多层次的复合替换中,``必须要额外的跳脱处理(反斜线),而$( )比较直观。
    3. $( )的弊端是,并不是所有的类unix系统都支持这种方式,反引号基本上可用在全部的 unix shell 中使用,若写成 shell script ,其移植性比较高。
    4. ``(反引号)需要进行转义。而$( )无需转义。

例子2:

# 将cmd1执行结果作为cmd2参数,再将cmd2结果作为cmd3的参数
cmd3 $(cmd2 $(cmd1))

# 如果是用反引号,直接引用是不行的,还需要作跳脱处理
cmd3 `cmd2 \`cmd1\``

例子3:

echo `echo ‘\\‘`
#输出:

echo $(echo ‘\\‘)
#输出
\\

例子4:

假设:

VS100COMNTOOLS=C:\Program Files\Microsoft Visual Studio 10.0\Common7\Tools\

在终端输入:

echo `echo "${VS100COMNTOOLS}" | sed ‘s#^.\+\\.\+#\1#‘`

输出

C:\Program Files\Microsoft Visual Studio 10.0\Common7\Tools\
echo $(echo "${VS100COMNTOOLS}" | sed ‘s#^.\+\\.\+#\1#‘)

输出

C:\Program Files\Microsoft Visual Studio 10.0\Common7

$(...)形式的命令替换达到了预期目的,而`...`却没有达到的原因是:

`...`将sed ‘...\\...‘中的\\转义成了sed ‘...\....‘

改成

echo `echo "${VS100COMNTOOLS}" | sed ‘s#^.\+\\\\.\+#\1#‘`

或者使用$(...)即可

三. 拓展

(1)${ }变量替换

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

$ A=B
$ echo ${A}B
#输出:
BB

(2)取路径、文件名、后缀

先赋值一个变量为一个路径,如: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

记忆方法如下:

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

(3)取子串及替换

命令             解释                        结果

${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

(4)根据状态为变量赋值

命令              解释                                 备注

${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 也受影响.

(5)数组

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]=xyz   则是将第四个组数重新定义为 xyz

(6)$(( ))与整数运算

bash中整数运算符号

符号         功能

+ - * /       分别为加、减、乘、除

%           余数运算

& | ^ !       分别为“AND、OR、XOR、NOT”

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

$ a=5;b=7;c=2
$ echo $((a+b*c))
19
$ echo $(($a+$b*$c))
19

(7)进制转换

$(( ))可以将其他进制转成十进制数显示出来。用法如下:

echo $((N#xx))

其中,N为进制,xx为该进制下某个数值,命令执行后可以得到该进制数转成十进制后的值。

$ echo $((2#110))   # 二进制转十进制
6
$ echo $((16#2a))   # 十六进制转十进制
42
$ echo $((8#11))    # 八进制转十进制
9

(7)(( ))重定义变量值

$ a=5;b=7
$ ((a++));echo $a
6
$ ((a--));echo $a
5
$ ((a<b));echo $?
0

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

时间: 2024-10-13 00:57:30

Shell中命令替换$(...)与后置引用`...`的区别与联系的相关文章

关于Shell中命令替换$(...)与后置引用`...`的不同

命令替换 在bash中,$( )与` `(反引号)都是用来作命令替换的.命令替换与变量替换差不多,都是用来重组命令行的,先完成引号里的命令行,然后将其结果替换出来,再重组成新的命令行. $( )与`` 在操作上,这两者都是达到相应的效果,但是建议使用$( ),理由如下: ``很容易与''搞混乱,尤其对初学者来说. 在多层次的复合替换中,``必须要额外的跳脱处理(反斜线),而$( )比较直观. 最后,$( )的弊端是,并不是所有的类unix系统都支持这种方式,但反引号是肯定支持的. # 将cmd1

shell中命令间的逻辑关系

在shell中命令执行成功返回值为0,不成功返回值为非0 逻辑与: shell中0为真,非0为假 第一个条件为真时,第二条件必须判断: 第一个条件为假时,第二条件不再判断,最终结果为假: 当第一个命令执行成功继续执行第二个命令,最终结果由第二个命令执行结果决定. 当第一个命令执行失败第二个命令不再执行,最终结果为失败. 理解为乘法运算(0为假1为真) 1*1=1        1*0=0          0*1=0        0*0=0 例: 1.user1存在,前一条命令执行成功,第二条

【转】Bash Shell中命令行选项/参数处理

原文网址:http://www.cnblogs.com/FrankTan/archive/2010/03/01/1634516.html 0.引言 写程序的时候经常要处理命令行参数,本文描述在Bash下的命令行处理方式. 选项与参数: 如下一个命令行: ./test.sh -f config.conf -v --prefix=/home 我们称-f为选项,它需要一个参数,即config.conf, -v 也是一个选项,但它不需要参数. --prefix我们称之为一个长选项,即选项本身多于一个字符

Bash Shell中命令行选项/参数处理

0.引言 写程序的时候经常要处理命令行参数,本文描述在Bash下的命令行处理方式. 选项与参数: 如下一个命令行: ./test.sh -f config.conf -v --prefix=/home 我们称-f为选项,它需要一个参数,即config.conf, -v 也是一个选项,但它不需要参数. --prefix我们称之为一个长选项,即选项本身多于一个字符,它也需要一个参数,用等号连接,当然等号不是必须的,/home可以直接写在--prefix后面,即--prefix/home,更多的限制后

shell中命令之间数据的传递

1.管道 "|" ls | cat -n > out.txt 2. 子shell 2.1 子shell 说明 在shell脚本中可以用()操作符可以定义一个子shell #/bin/bash pwd (cd / ; ls) pwd 执行结果 说明: 当命令在子shell中执行的时候不会对当前shell有任何影响 2.2 获取子shell 中的数据 cmd_output=$(COMMANDS) out=$(ls -l | cat -n) #!/bin/bash out=$(ls -

shell中命令的执行流程

在shell中,一个命令有3中写法: 1 可以直接写(Normal Command) 2 可以放在双引号中("Command") 3 可以放在单引号中('Comand') 这3中写法,命令的执行流程也是不一样的,如下图所示: alias替换 将alias替换成真正的命令,比如将ls替换成ls --color=auto brace替换 将a{b,c}e替换成abe,ace ~替换 将~替换成HOME目录 变量替换 将$开头的变量替换为其真正的值 执行替换命令 即执行$(command)或

shell中的替换

shell中如果存在一些特殊的字符,就需要进行替换,可进行命令替换.变量替换.转义替换 1.转义字符的替换 shell中包含以下的转移字符 \a   响铃警报\\   反斜杠 \b  退格(删除键) \f   换页(FF),将当前位置移到下页开头 \n  换行 \r   回车 \t   水平制表符 \v   垂直制表符 以上转移字符都可以使用在echo中,通过使用-e 完成转义字符的替换,否者会原样输出,也可通过-E 禁止转移,通过-n可以禁止插入换行符 2.命令替换 如下声明将命令date赋值

C++中前置操作符和后置操作符的重载

1,C 语言中讨论了原生含义,C++ 中有必要考虑前置.后置操作符能够重载,有何问题: 2,值得思考的问题: 1,下面的代码有没有区别?为什么? 1,i++  // i 的值作为返回值,i 自增 1: 2,++i  // i 自增 1,i 的值作为返回值: 3,没有使用返回值,由于编译器(不同的编译器都是一样的)的优化,在工程上面,这两行代码没有区别: 2,真的有区别吗?编程实验: 1,main.cpp 文件: 1 #include <iostream> 2 #include <stri

shell中的单引号和双引号的区别

对于常规的字符串定义变量值应添加双引号,并且等号后不能 有空格,需要强引用的,则需要单引号,需要命令引用的使用反引号. 单引号:所见即所得,即输出时会将单引号内的所有内容都原样输出,或者描述为单引号里面看到的什么就输出什么,这成为强引用. 双引号:输出双引号的所有内容:如果内容中有命令(要反引).变量.特殊转义,会先把变量.命令.转义字符解析出结果,然后在输出最终内容,这称为弱引. 反引号:一般用于命令,执行的时候命令会被执行,相当于$(),赋值和输出都要用反引号引起来. [[email pro