Linux Shell exec绑定重定向

shell 中的 exec 两种用法

1.exec 命令 ;命令代替shell程序,命令退出,shell 退出;比如 exec ls

  2.exec 文件重定向,可以将文件的重定向就看为是shell程序的文件重定向 比如 exec 5</dev/null;exec 5<&-

shell的内建命令exec将并不启动新的shell,而是用要被执行命令替换当前的shell进程,并且将老进程的环境清理掉,而且exec命令后的其它命令将不再执行。
  因此,如果你在一个shell里面,执行exec ls那么,当列出了当前目录后,这个shell就自己退出了,因为这个shell进程已被替换为仅仅执行ls命令的一个进程,执行结束自然也就退出了。为了避免这个影响我们的使用,一般将exec命令放到一个shell脚本里面,用主脚本调用这个脚本,调用点处可以用bash a.sh,(a.sh就是存放该命令的脚本),这样会为a.sh建立一个sub shell去执行,当执行到exec后,该子脚本进程就被替换成了相应的exec的命令。
  source命令或者".",不会为脚本新建shell,而只是将脚本包含的命令在当前shell执行。
  不过,要注意一个例外,当exec命令来对文件描述符操作的时候,就不会替换shell,而且操作完成后,还会继续执行接下来的命令。
  exec 3<&0:这个命令就是将操作符3也指向标准输入。

  另外,这个命令还可以作为find命令的一个选项,如下所示:
  (1)在当前目录下(包含子目录),查找所有txt文件并找出含有字符串"bin"的行
    find ./ -name “.txt" -exec grep “bin” {} ;
  (2)在当前目录下(包含子目录),删除所有txt文件
    find ./ -name ".txt” -exec rm {} ;
先总结一个表:

exec命令作用
exec ls在shell中执行ls,ls结束后不返回原来的shell中了
exec <file将file中的内容作为exec的标准输入
exec >file将file中的内容作为标准写出
exec 3<file将file读入到fd3中
sort <&3fd3中读入的内容被分类
exec 4>file将写入fd4中的内容写入file中
ls >&4Ls将不会有显示,直接写入fd4中了,即上面的file中
exec 5<&4创建fd4的拷贝fd5
exec 3<&-关闭fd3

  1、exec 执行程序  

虽然exec和source都是在父进程中直接执行,但exec这个与source有很大的区别,source是执行shell脚本,而且执行后会返回以前的shell。而exec的执行不会返回以前的shell了,而是直接把以前登陆shell作为一个程序看待,在其上经行复制。

  举例说明:

:~/test#exec ls

exp1 exp5 linux-2.6.27.54 ngis_post.sh test xen-3.0.1-install

:~/test#exec >text

:~/test#ls

:~/test#pwd

:~/test#echo “hello”

:~/test#exec>/dev/tty

:~/test#cat text

exp1

exp5

linux-2.6.27.54

ngis_post.sh

test

text

xen-3.0.1-install

/root/test

hello

:~/test#

Exec >text 是将当前shell的标准输出都打开到text文件中

:~/test#cat test

ls

Pwd

:~/test#bash

:~/test#exec <test

:~/test#ls

exp1 exp5 linux-2.6.27.54 ngis_post.sh test text xen-3.0.1-install

:~/test#pwd

/root/test

:~/test#

:~/test#exit #自动执行

  exec的重定向

先上我们进如/dev/fd/目录下看一下:

:~/test#cd /dev/fd

:/dev/fd#ls

0 1 2 255

  默认会有这四个项:0是标准输入,默认是键盘。

  1是标准输出,默认是屏幕/dev/tty

  2是标准错误,默认也是屏幕

  255

  当我们执行exec 3>test时:

:/dev/fd#exec 3>/root/test/test

:/dev/fd#ls

0 1 2 255 3

:/dev/fd#

看到了吧,多了个3,也就是又增加了一个设备,这里也可以体会下linux设备即文件的理念。这时候fd3就相当于一个管道了,重定向到fd3中的文件会被写在test中。关闭这个重定向可以用exec 3>&-。

:/dev/fd#who >&3

:/dev/fd#ls >&3

:/dev/fd#exec 3>&-

:/dev/fd#cat /root/test/te

test text

:/dev/fd#cat /root/test/test

root tty1 2010-11-16 01:13

root pts/0 2010-11-15 22:01 (192.168.0.1)

root pts/2 2010-11-16 01:02 (192.168.0.1)

0

1

2

255

3

应用举例

1 exec 3<test
2 
3 while read -u 3 pkg

  do

  echo “$pkg”

  done

  系统调用exec是以新的进程去代替原来的进程,但进程的PID保持不变。因此,可以这样认为,exec系统调用并没有创建新的进程,只是替换了原来进程上下文的内容。原进程的代码段,数据段,堆栈段被新的进程所代替。

  一个进程主要包括以下几个方面的内容:

  (1)一个可以执行的程序

  (2) 与进程相关联的全部数据(包括变量,内存,缓冲区)

  (3)程序上下文(程序计数器PC,保存程序执行的位置)

exec是一个函数簇,由6个函数组成,分别是以excl和execv打头的。
  执行exec系统调用,一般都是这样,用fork()函数新建立一个进程,然后让进程去执行exec调用。我们知道,在fork()建立新进程之后,父进各与子进程共享代码段,但数据空间是分开的,但父进程会把自己数据空间的内容copy到子进程中去,还有上下文也会copy到子进程中去。而为了提高效率,采用一种写时copy的策略,即创建子进程的时候,并不copy父进程的地址空间,父子进程拥有共同的地址空间,只有当子进程需要写入数据时(如向缓冲区写入数据),这时候会复制地址空间,复制缓冲区到子进程中去。从而父子进程拥有独立的地址空间。而对于fork()之后执行exec后,这种策略能够很好的提高效率,如果一开始就copy,那么exec之后,子进程的数据会被放弃,被新的进程所代替。

  exec与system的区别
  (1) exec是直接用新的进程去代替原来的程序运行,运行完毕之后不回到原先的程序中去。

  (2) system是调用shell执行你的命令,system=fork+exec+waitpid,执行完毕之后,回到原先的程序中去。继续执行下面的部分。

  总之,如果你用exec调用,首先应该fork一个新的进程,然后exec. 而system不需要你fork新进程,已经封装好了。