source/exec/./env命令区别

来源:互联网 发布:小米max网络参数 编辑:程序博客网 时间:2024/05/16 07:19
source命令用法:

  source FileName

  作用:在当前bash环境下读取并执行FileName中的命令。

注:该命令通常用命令“.”来替代。 

  如:source .bash_rc 与 . .bash_rc 是等效的。

  source命令(从 C Shell 而来)是bash shell的内置命令。点命令,就是个点 符号,(从Bourne

  Shell而来)是source的另一名称。同样的,当前脚本中配置的变量也将作为脚 本的环境,source(或点)命令通常用于重新执行刚 修改的初始化

  文档,如 .bash_profile 和 .profile 等等。例如,假如在登录后对 .bash_profile 中的 EDITER 和

  TERM 变量做了修改,则能够用source命令重新执行 .bash_profile 中的命令 而不用注销并重新登录。

  比如您在一个脚本里export $KKK=111 ,假如您用./a.sh执行该脚本,执行完 毕后,您运行 echo $KKK

  ,发现没有值,假如您用source来执行 ,然后再echo

  ,就会发现KKK=111。因为调用./a.sh来执行shell是在一个子shell里运行的, 所以执行后,结构并没有反应到父shell 里,但是

  source不同他就是在本shell中执行的,所以能够看到结果


  source命令(从 C Shell 而来)是bash shell的内置命令。点命令,就是一个 点符号,(从Bourne

  Shell而来)是source的另一名称。这两个命令都以一个脚本为参数,该脚本将 作为当前shell的环境执行,即不会启动一个新的子进 程。所有在脚

  本中设置的变量将成为当前Shell的一部分。同样的,当前脚本中设置的变量 也将作为脚本的环境,source(或点)命令通常用于重新执行 刚修改的初始

  化文件,如 .bash_profile 和 .profile 等等。例如,如果在登录后对 .bash_profile 中的 EDITER 和

  TERM 变量做了修改,则可以用source命令重新执行 .bash_profile 中的命令 而不用注销并重新登录。象

  .bash_profile 或其它类似的Shell脚本这样,文件无需可执行权限即可用 source或点命令执行。

  source命令的一个妙用

  在编译核心时,常常要反复输入一长串命令,如

  make mrproper

  make menuconfig

  make dep

  make clean

  make bzImage

  .......

  这些命令既长,又繁琐。而且有时候容易输错,浪费你的时间和精力。如果把 这些命令做成一个文件,让它自动按顺序执行,对于需要多次反复编译核 心的用

  户来说,会很方便。用source命令可以办到这一点。它的作用就是把一个文件 的内容当成是shell来执行。先在/usr/src /linux-

  2.4.20目录下建立一个文件,取名为make_command:

  在其中输入如下内容:

  make mrproper &&

  make menuconfig &&

  make dep &&

  make clean &&

  make bzImage &&

  make modules &&

  make modules_install &&

  cp arch/i386/boot/bzImge /boot/vmlinuz_new &&

  cp System.map /boot &&

  vi /etc/lilo.conf &&

  lilo -v

  文件建立好之后,以后每次编译核心,只需要在/usr/src/linux-2.4.20下输入

  source make_command

  就行了。这个文件也完全可以做成脚本,只需稍加改动即可。这里主要是让大 家理解source的用法。如果你用的不是lilo来引导系统,可以 把最后两句话 去 掉。配置你自己的引导程序来引导新内核。

  shell编程中的命令有时和C语言是一样的。&&表示与,||表示或。把两个命令 用&&联接起来,如

  make mrproper && make menuconfig

  ,表示要第一个命令执行成功才能执行第二个命令。对执行顺序有要求的命令 能保证一旦有错误发生,下面的命令不会盲目地继续执行

  exec:

   在bash下输入man exec,找到exec命令解释处,可以看到有”No new process is created.”这样的解释,这就是说exec命令不产生新的子进程。那么exec与source的区别是什么呢?
exec命令在执行时会把当前的shell process关闭,然后换到后面的命令继续执行。因此,如果你在一个shell里面,执行exec ls那么,当列出了当前目录后,这个shell就自己退出了,因为这个shell进程已被替换为仅仅执行ls命令的一个进程,执行结束自然也就退出了。为了避免这个影响我们的使用,一般将exec命令放到一个shell脚本里面,用主脚本调用这个脚本,调用点处可以用bash a.sh,(a.sh就是存放该命令的脚本),这样会为a.sh建立一个sub shell去执行,当执行到exec后,该子脚本进程就被替换成了相应的exec的命令。
 
1. 系统调用exec是以新的进程去代替原来的进程,但进程的PID保持不变。因此,可以这样认为,exec系统调用并没有创建新的进程,只是替换了原来进程上下文的内容。原进程的代码段,数据段,堆栈段被新的进程所代替。
一个进程主要包括以下几个方面的内容:
(1)一个可以执行的程序
(2) 与进程相关联的全部数据(包括变量,内存,缓冲区)
(3)程序上下文(程序计数器PC,保存程序执行的位置)
 
2. exec是一个函数簇,由6个函数组成,分别是以excl和execv打头的。
执行exec系统调用,一般都是这样,用fork()函数新建立一个进程,然后让进程去执行exec调用。我们知道,在fork()建立新进程之后,父进各与子进程共享代码段,但数据空间是分开的,但父进程会把自己数据空间的内容copy到子进程中去,还有上下文也会copy到子进程中去。而为了提高效率,采用一种写时copy的策略,即创建子进程的时候,并不copy父进程的地址空间,父子进程拥有共同的地址空间,只有当子进程需要写入数据时(如向缓冲区写入数据),这时候会复制地址空间,复制缓冲区到子进程中去。从而父子进程拥有独立的地址空间。而对于fork()之后执行exec后,这种策略能够很好的提高效率,如果一开始就copy,那么exec之后,子进程的数据会被放弃,被新的进程所代替。
 
3. exec与system的区别
(1) exec是直接用新的进程去代替原来的程序运行,运行完毕之后不回到原先的程序中去。
(2) system是调用shell执行你的命令,system=fork+exec+waitpid,执行完毕之后,回到原先的程序中去。继续执行下面的部分。
总之,如果你用exec调用,首先应该fork一个新的进程,然后exec. 而system不需要你fork新进程,已经封装好了。

exec I/O重定向

1、 基本概念

a、 I/O重定向通常与 FD有关,shell的FD通常为10个,即 0~9;
b、 常用FD有3个,为0(stdin,标准输入)、1(stdout,标准输出)、2(stderr,标准错误输出),默认与keyboard、monitor、monitor有关;
c、 在IO重定向中,stdout与stderr的管道会先准备好,才会从stdin 读进资料;
d、 管道“|”(pipe line):上一个命令的 stdout 接到下一个命令的 stdin;
e、 tee 命令是在不影响原本 I/O 的情况下,将 stdout 复制一份到档案去;
f、 bash(ksh)执行命令的过程:分析命令-变量求值-命令替代(``和$( ))-重定向-通配符展开-确定路径-执行命令;
g、 ( ) 将 command group 置于 sub-shell 去执行,也称 nested sub-shell,它有一点非常重要的特性是:继承父shell的Standard input, output, and error plus any other open file descriptors。
h、 exec 命令:常用来替代当前 shell 并重新启动一个 shell,换句话说,并没有启动子 shell。使用这一命令时任何现有环境都将会被清除。exec 在对文件描述符进行操作的时候,也只有在这时,exec 不会覆盖你当前的 shell 环境。

(1)、cmd &n 使用系统调用 dup (2) 复制文件描述符 n 并把结果用作标准输出
&- 关闭标准输出
n&- 表示将 n 号输出关闭
上述所有形式都可以前导一个数字,此时建立的文件描述符由这个数字指定而不是缺省的 0 或 1。如:
... 2>file 运行一个命令并把错误输出(文件描述符 2)定向到 file。
... 2>&1 运行一个命令并把它的标准输出和输出合并。(严格的说是通过复制文件描述符 1 来建立文件描述符 2 ,但效果通常是合并了两个流。)
我们对 2>&1详细说明一下 :2>&1 也就是 FD2=FD1 ,这里并不是说FD2 的值 等于FD1的值,因为 > 是改变送出的数据信道,也就是说把 FD2 的 “数据输出通道” 改为 FD1 的 “数据输出通道”。如果仅仅这样,这个改变好像没有什么作用,因为 FD2 的默认输出和 FD1的默认输出本来都是 monitor,一样的!
但是,当 FD1 是其他文件,甚至是其他 FD 时,这个就具有特殊的用途了。请大家务必理解这一点。
(2)、 如果 stdin, stdout, stderr 进行了重定向或关闭, 但没有保存原来的 FD, 可以将其恢复到 default 状态吗?
*** 如果关闭了stdin,因为会导致退出,那肯定不能恢复。
*** 如果重定向或关闭 stdout和stderr其中之一,可以恢复,因为他们默认均是送往monitor(但不知会否有其他影响)。如恢复重定向或关闭的 stdout: exec 1>&2 ,恢复重定向或关闭的stderr:exec 2>&1。
*** 如果stdout和stderr全部都关闭了,又没有保存原来的FD,可以用:exec 1>/dev/tty 恢复。
(3)、 cmd >a 2>a 和 cmd >a 2>&1 为什么不同?
cmd >a 2>a :stdout和stderr都直接送往文件 a ,a文件会被打开两遍,由此导致stdout和stderr互相覆盖。
cmd >a 2>&1 :stdout直接送往文件a ,stderr是继承了FD1的管道之后,再被送往文件a 。a文件只被打开一遍,就是FD1将其打开。

2.exec命令使用

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

(1). exec 执行程序

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

(2). exec的重定向

先上我们进如/dev/fd/目录下看一下:
root@localhost :~/test#cd /dev/fd
root@localhost :/dev/fd#ls
0  1  2 255
默认会有这四个项:0是标准输入,默认是键盘。
1是标准输出,默认是屏幕/dev/tty
2是标准错误,默认也是屏幕
当我们执行exec 3>test时:
root@localhost :/dev/fd#exec 3>/root/test/test 
root@localhost :/dev/fd#ls
0  1  2  255  3
root@localhost :/dev/fd#

看到了吧,多了个3,也就是又增加了一个设备,这里也可以体会下linux设备即文件的理念。这时候fd3就相当于一个管道了,重定向到fd3中的文件会被写在test中。关闭这个重定向可以用exec 3>&-。
root@localhost :/dev/fd#who >&3
root@localhost :/dev/fd#exec 3>&-
root@localhost :/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)

  别处,这个命令还可以作为find命令的一个选项,如下所示:

  (1)在当前目录下(包含子目录),查找所有txt文件并找出含有字符串"bin"的行

  find ./ -name "*.txt" -exec grep "bin" {}

  (2)在当前目录下(包含子目录),删除所有txt文件

  find ./ -name "*.txt" -exec rm {}

env:
env - run a program in a modified environment
0 0