shell教程
OutOfMemory.CN技术专栏-> shell-> shell教程-> 第22章. 进程替换

第22章. 进程替换

进程替换与命令替换(commandsubstitution)很相似.命令替换把一个命令的结果赋给一个变量,例如dir_contents=`ls-al`或xref=$(grepworddatafile).进程替换则是把一个进程的输出回馈给另一个进程(换句话说,它把一个命令的结果发送给另一个命令).命令替换的一般形式由圆括号括起的命令>(command)<(command)启动进程替换

进程替换命令替换(command substitution)很相似. 命令替换把一个命令的结果赋给一个变量,例如 dir_contents=`ls -al`xref=$( grep word datafile). 进程替换则是把一个进程的输出回馈给另一个进程 (换句话说,它把一个命令的结果发送给另一个命令).

命令替换的一般形式

由圆括号括起的命令

>(command)

<(command)

启动进程替换. 它是用/dev/fd/<n>文件把在圆括号内的进程的处理结果发送给另外一个进程. [1] (译者注:实际上现代的UNIX类操作系统提供的/dev/fd/n文件是与文件描述相关的,整数n指的就是在进程运行时对应数字的文件描述符)

"<" 或or ">" 与圆括号之间是没有空格的. 如果加了空格将会引起错误信息.

 bash$ echo >(true)
 /dev/fd/63
 
 bash$ echo <(true)
 /dev/fd/63
 	      
Bash在两个文件描述符(file descriptors)之间创建了一个管道, --fInfOut--. true命令的标准输入被连接到fOut(dup2(fOut, 0)), 然后Bash把/dev/fd/fIn作为参数传给echo.如果系统的/dev/fd/<n>文件不够时,Bash会使用临时文件. (Thanks, S.C.)

进程替换能比较两个不同命令之间的输出,或者甚至相同命令不同选项的输出.

 bash$ comm <(ls -l) <(ls -al)
 total 12
-rw-rw-r--    1 bozo bozo       78 Mar 10 12:58 File0
-rw-rw-r--    1 bozo bozo       42 Mar 10 12:58 File2
-rw-rw-r--    1 bozo bozo      103 Mar 10 12:58 t2.sh
        total 20
        drwxrwxrwx    2 bozo bozo     4096 Mar 10 18:10 .
        drwx------   72 bozo bozo     4096 Mar 10 17:58 ..
        -rw-rw-r--    1 bozo bozo       78 Mar 10 12:58 File0
        -rw-rw-r--    1 bozo bozo       42 Mar 10 12:58 File2
        -rw-rw-r--    1 bozo bozo      103 Mar 10 12:58 t2.sh

用进程替换来比较两个不同目录的内容 (考察哪些文件名是相同的,哪些是不同的):

diff <(ls $first_directory) <(ls $second_directory)

其他一些进程替换的用法和技巧:

cat <(ls -l)
# 等同于     ls -l | cat

sort -k 9 <(ls -l /bin) <(ls -l /usr/bin) <(ls -l /usr/X11R6/bin)
# 列出系统中3个主要的'bin'目录的所有文件,并且按文件名排序.
# 注意是三个明显不同的命令输出回馈给'sort'.


diff <(command1) <(command2)    # 给出两个命令输出的不同之处.

tar cf >(bzip2 -c > file.tar.bz2) $directory_name
# 调用"tar cf /dev/fd/?? $directory_name",和"bzip2 -c > file.tar.bz2".
#
# 因为/dev/fd/<n>的系统属性,
# 所以两个命令之间的管道不必是命名的.
#
# 这种效果可以模仿出来.
#
bzip2 -c < pipe > file.tar.bz2&
tar cf pipe $directory_name
rm pipe
#        或者
exec 3>&1
tar cf /dev/fd/4 $directory_name 4>&1 >&3 3>&- | bzip2 -c > file.tar.bz2 3>&-
exec 3>&-


# Thanks, Sthane Chazelas

有个读者给我发来下面关于进程替换的有趣例子A.

# 摘自SuSE发行版中的代码片断:

while read  des what mask iface; do
# 这里省略了一些命令 ...
done < <(route -n)


# 为了测试它,我们来做些动作.
while read  des what mask iface; do
  echo $des $what $mask $iface
done < <(route -n)

# 输出:
# Kernel IP routing table
# Destination Gateway Genmask Flags Metric Ref Use Iface
# 127.0.0.0 0.0.0.0 255.0.0.0 U 0 0 0 lo



# 由 Sthane Chazelas给出的,一个更容易理解的等价代码是:
route -n |
  while read des what mask iface; do   # 管道的输出被赋给了变量.
    echo $des $what $mask $iface
  done  #  这样就取回了和上面一样的输出.
        #  但是, Ulrich Gayer指出 . . .
        #+ 这个简单版本的等价代码在while循环中使用了一个子shell,
        #+ 因此当管道结束后变量会被毁掉.



#  更进一步, Filip Moritz解释了上面两个例子之间有一个细微的不同之处
#+ 如下所示.

(
route -n | while read x; do ((y++)); done
echo $y # $y 仍然没有被声明或设置

while read x; do ((y++)); done < <(route -n)
echo $y # $y的值为 route -n 输出的行数
)

# 一般来说
(
: | x=x
# 看上去是启动了一个子shell
: | ( x=x )
# 但
x=x < <(:)
# 实际上不是
)

# 当解析csv或类似的东西时非常有用.
# 事实上,这就是SuSE原本的代码片断所要实现的功能.

[1]

这与命名管道(named pipe)(临时文件)有相同的作用, 事实上命名管道同样在进程替换中被使用.

© 内存溢出 OutOfMemory.CN