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

Linux Shell编程(5)——shell特殊字符(下)

2014-09-30 07:18 363 查看
{}

代码块[花括号]. 这个结构也是一组命令代码块,事实上,它是匿名的函数。然而与一个函数所不同的,在代码块里的变量仍然能被脚本后面的代码访问。

bash$ { local a;
a=123; }
bash: local: can only be used in a
function

a=123
{ a=321; }
echo "a = $a" # a = 321 (结果是在代码块里的值)

由花括号括起的代码块可以引起输入输出的I/O重定向。

#!/bin/bash
# 从/etc/fstab文件里按一次一行地读.

File=/etc/fstab

{
read line1
read line2
} < $File

echo "First line in $File is:"
echo "$line1"
echo
echo "Second line in $File is:"
echo "$line2"

exit 0

# 现在,你如何解析每一行的分割符?
# 提示: 使用awk.

把一个代码块的结果写进一个文件

#!/bin/bash
# rpm-check.sh

# 查询一个rpm安装包的描述,软件清单,和是否它能够被安装.
# 并把结果保存到一个文件中.
#
# 这个脚本使用一个代码块来举例说明。

SUCCESS=0
E_NOARGS=65

if [ -z "$1" ]
then
echo "Usage: `basename $0` rpm-file"
exit $E_NOARGS
fi

{
echo
echo "Archive Description:"
rpm -qpi $1 # 查询软件包的描述.
echo
echo "Archive Listing:"
rpm -qpl $1 # 查询软件包中的软件清单.
echo
rpm -i --test $1 # 查询该软件包能否被安装.
if [ "$?" -eq $SUCCESS ]
then
echo "$1 can be installed."
else
echo "$1 cannot be installed."
fi
echo
} > "$1.test" # 把代码块的所有输出重定向到一个文件中。

echo "Results of rpm test in file $1.test"

# 参考rpm的man手册来理解上面所用的选项。

exit 0

不像一个用圆括号括起来的命令组,一个用花括号括起的代码块不会以一个子shell运行。[2]

{} \;
路径名. 基本上用于find命令里.它不是shell内建的.

分号";"结束find命令中-exec选项的命令序列.它应该转义一下以免被shell误解释。

[ ]
测试.

测试在[ ]中的表达式. 注意[是shell内建的测试的一部分(同义于测试),并非 是外部命令/usr/bin/test的链接.

[[ ]]
测试.

测试[[ ]]之中的表达式(shell的关键字).

参考[[ ... ]]结构的讨论.

[ ]
数组元素.

在数组的上下文中,方括号表示数组的每个元素的数字编号.
1 Array[1]=slot_1
2 echo ${Array[1]}

[ ]
字符集的范围.

用于正则表达式的一部分,方括号描述一个匹配的字符集范围.

(( ))
整数扩展.

扩展并计算(( ))里的整数表达式[译者注:粗心的读者要注意了,是整数计算,可不能用来做浮点计算].

参考(( ... ))结构的讨论.

> &> >& >> <
redirection重定向.

scriptname >filename重定向scriptname的输出到文件filename中去. 如果文件filename存在则将会被覆盖.

command &>filename 会重定向命令command标准输出(stdout)和标准错误(stderr)到文件filename中.

command >&2 把命令command的标准输出(stdout)重定向到标准错误(stderr).

scriptname >>filename appends把脚本scriptname的输出追加到文件filename.如果filename不存在,则它会被创建.

进程替换.

(command)>

<(command)

在不同的上下文中, 字符 "<"和">"会被当作字符比较操作符.

在另一种不同的上下文中, 字符"<"和">"被当作整数比较操作符. 请参考例子 12-9.

<<
在here document中使用的重定向.

<<<
在 here string中使用的重定向.

<, >
ASCII比较.
veg1=carrots
veg2=tomatoes

if [[ "$veg1" < "$veg2" ]]
then
echo "Although $veg1 precede $veg2 in the dictionary,"
echo "this implies nothing about my culinary preferences."
else
echo "What kind of dictionary are you using, anyhow?"
fi

\<, \>
在 regular expression中的单词界线.

bash$ grep '\<the\>' textfile

|

管道. 把上一个命令的输出传给下一个命令,或是shell. 这是连接命令的一种方法.

echo ls -l | sh
# 把"echo ls -l"的输出传给shell,
#+ 这等同与直接的"ls -l".

cat *.lst | sort | uniq
# 合并且排序所有的".lst"文件,然后删除多余的相同行.

经典的进程间通信方法之一的管道能把一个进程的标准输出发到另一个进程的标准输入.在一种典型的情况,一个命令,比如说cat或echo, 管道传递一个数据流给过滤器(一个改变了它的输入的命令)。

cat $filename1 $filename2 | grep $search_word

一个命令或一组命令的输出可以由管道传给一个脚本.
#!/bin/bash
# uppercase.sh : 把输入字符改为大写.

tr 'a-z' 'A-Z'
# 字母的范围一定要引号引起来,
#+ 这样才能保护文件名而不会被扩展成单个字母的文件名.

exit 0
现在,让我们把ls -l的输出用管道与这个脚本连起来.
bash$ ls -l | ./uppercase.sh
-RW-RW-R-- 1 BOZO BOZO 109 APR 7 19:49 1.TXT
-RW-RW-R-- 1 BOZO BOZO 109 APR 14 16:48 2.TXT
-RW-R--R-- 1 BOZO BOZO 725 APR 20 20:56 DATA-FILE

管道里的每一个进程的标准输出都被当成下一个命令的标准输入. 如果不是这种情况,那么数据流会阻塞,并且管道不会引起预期的效果。
cat file1 file2 | ls -l | sort
# 来自"cat file1 file2"的输出会消失.

管道以子进程来运行, 因此不能引起变量的改变。
variable="initial_value"
echo "new_value" | read variable
echo "variable = $variable" # variable = initial_value

如果在管道中的一个命令失败了,会过早的终结整个管道的执行。这称为管道破坏(broken pipe),这时会发送一个叫SIGPIPE 的信号.

>|
强迫重定向 (即使noclobber选项设置). 这会强迫覆盖一个存在的文件.

||
或逻辑操作符. 在一个测试结构中,||操作符当测试条件的任何一个为真时返回0 (成功)的标志.

&
在后台运行作业. 一个后面跟一个&的命令会在后台运行.
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: