您的位置:首页 > 运维架构 > Shell

Shell脚本eval、``和$()、[[和[、 $(( ))和(())、${}

2016-06-15 17:19 253 查看
eval
eval的作用是再次执行命令行处理,也就是说,对一个命令行,执行两次命令行处理。

语法:eval cmdLine
eval会对后面的 cmdLine 进行两遍扫描,如果第一遍扫描后 cmdLine 是个普通命令,则执行此命令;
如果 cmdLine 中含有变量的间接引用,则保证间接引用的语义。

eval echo \$$# 取得最后一个参数
如:cat last
eval echo \$$#
./last one teo three four
four
第一遍扫描后,shell把反斜杠去掉了。当shell再次扫描该行时,它替换了$4的值,并执行echo命令

``(反引号)和$()
在bash shell中,$( )与` `(反引号)都是用来做命令替换(command substitution)用的。
$ echo the last sunday is $(date -d "last sunday" +%Y-%m-%d)
得到上一星期天的日期

用$( )的理由
1. ` `很容易与' '(单引号)搞混。有时在一些奇怪的字形显示中,两种符号是一模一样的(直竖两点)。
2. 在多层次的复合替换中,` `须要额外的跳脱(\`)处理,而$( )则比较直观。例如:
command1 `command2 `command3` `
原本的意图是在command2 `command3`中先将command3替换出来给command2处理,然后再将结果传给command1 `command2 ...`来处理。
然而,真正的结果在命令行中却是分成了`command2`与` `两段。
正确的输入应该如下:
command1 `command2 \`command3\` `
换成$( )则一目了然:
command1 $(command2 $(command3))
$( )的不足
` `基本上可在全部的unix shell中使用,若写成shell script移植性比较高。而$( )并不是每一种shell都能使用。

[[和[
1."[[",是关键字,许多shell(如ash bsh)并不支持这种方式。ksh, bash(据说从2.02起引入对[[的支持)等支持。
"["是一条命令, 与test等价,大多数shell都支持。在现代的大多数sh实现中,"["与"test"是内部(builtin)命令。
2.二者都支持算术比较和字符串比较表达式(具体使用可能有点不同)
(1)"-gt", "-lt"是算术比较操作符,用于比较整数的大小。
(2)">", "<"是字符串比较操作符,用于比较字符串的大小,使用字典顺序,与当前的locale有关。
(3)建议在使用数值比较的时候,使用let,(())命令,否则容易出错。

[ 和 [[ 不同比较:






由上面两张图可看出,[在进行字符比较的时候需要使用转义字符\,而[[不需要

3.逻辑与和逻辑或(1)"[":逻辑与:"-a";逻辑或:"-o";(2)"[[":逻辑与:"&&";逻辑或:"||"

4.
1)[ ... ]为shell命令,所以在其中的表达式应是它的命令行参数,所以串比较操作符">" 与"<"必须转义,否则就变成IO重定向了;(2)由于"[["是关键字,不会做命令行扩展,所以在[[中"<"与">"不需转义,但是相对的语法就稍严格些。例如在[ ... ]中可以用引号括起操作符,因为在做命令行扩展时会去掉这些引号,而在` `.``.``.` `则不允许这样做;



5.` `.``.``.` `进行算术扩展,而[ ... ]不做




$(( ))
用来作整数运算。在 bash 中,$(( ))的整数运算符号大致有这些:
+ - * / 加、减、乘、除
% 余数运算
& | ^ ! AND、OR、XOR、NOT运算
举例:
$ a=5; b=7; c=2
$ echo $((a+b*c))
19
$ echo $(((a+b)/c))
6
$ echo $(((a*b)%c))
1
在$(( ))中的变量名称也可以在其前面加 $ 符号:$(($a+$b*$c))也可以得到 19 的结果。
此外,$(( ))还可以作不同进制(如二进制、八进位、十六进制)运算,只是输出结果皆为十进制而已。
echo $((16#2a)) 结果为 42 (16进位转十进制)

举一个实用的例子:
当前的 umask 是 022,新建文件的权限为:
$ umask 022
$ echo "obase=8; $(( 8#666 & (8#777 ^ 8#$(umask)) ))" | bc
644

事实上,单纯用(( ))也可以重定义变量值,或作testing:
a=5; ((a++)) 将 $a 重定义为 6
a=5; ((a–)) a=4
a=5; b=7; ((a < b)) 会得到 0 (true) 的返回值
常见的用于(( ))的测试符号有以下这些:
< 小于
> 大于
<= 小于或等于
>= 大于或等于
== 等于
!= 不等于

${ }它其实就是用来作变量替换用的。一般情况下,$var 与 ${var} 并没有啥不一样。但是用 ${ } 会比较精确的界定变量名称的范围。
再来看${}的一些特异功能
定义一个变量: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:4}:提取最左边的 4 个字节:dir1${file:4:5}:提取第 54个字节右边的连续 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} :变量值为26个字节

test.sh:
file=dir1/dir2/dir3/my.file.txt

echo ${file#*/}
echo ${file##*/}
echo ${file#*.}
echo ${file##*.}
echo ${file%/*}
echo ${file%%/*}
echo ${file%.*}
echo ${file%%.*}

echo ${file:0:4}
echo ${file:4:5}

echo ${file/dir/path}
echo ${file//dir/path}

echo ${#file}
运行结果:




以上内容部分转载
《完》
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  eval