9 exec linux_exec linux - CSDN
  • linux下的exec命令

    2018-08-29 18:37:31
    shell 中的 exec 两种用法: 1.exec 命令 ;命令代替shell程序,命令退出,shell 退出;比如 exec ls 2.exec 文件重定向,可以将文件的重定向就看为是shell程序的文件重定向 比如 exec 5</dev/null;exec 5&...

    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 <&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作为一个程序看待,在其上经行复制。


    举例说明:


    root@localhost:~/test#exec ls


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


    <logout>


     


    root@localhost:~/test#exec >text


    root@localhost:~/test#ls


    root@localhost:~/test#pwd


    root@localhost:~/test#echo "hello"


    root@localhost:~/test#exec>/dev/tty


    root@localhost:~/test#cat text 


    exp1


    exp5


    linux-2.6.27.54


    ngis_post.sh


    test


    text


    xen-3.0.1-install


    /root/test


    hello


    root@localhost:~/test#


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


     


    root@localhost:~/test#cat test


    ls


    Pwd


    root@localhost:~/test#bash


    root@localhost:~/test#exec <test


    root@localhost:~/test#ls


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


    root@localhost:~/test#pwd


    /root/test


    root@localhost:~/test#


    root@localhost:~/test#exit       #自动执行


     


    2. exec的重定向


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


    root@localhost:~/test#cd /dev/fd


    root@localhost:/dev/fd#ls


    0  1  2  255


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


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


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


    255


    当我们执行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#ls >&3


    root@localhost:/dev/fd#exec 3>&-


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


    test  text  


    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)


    0


    1


    2


    255


    3


     


     


     


     


    3. 应用举例:


            exec 3<test


            while read -u 3 pkg


    do


    echo "$pkg"


    done




    . 系统调用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、 用  来改变送出的数据信道(stdout, stderr),使之输出到指定的档案;


    e、 0 是  与 1> 是一样的;


    f、 在IO重定向 中,stdout 与 stderr 的管道会先准备好,才会从 stdin 读进资料;


    g、 管道“|”(pipe line):上一个命令的 stdout 接到下一个命令的 stdin;


    h、 tee 命令是在不影响原本 I/O 的情况下,将 stdout 复制一份到档案去;


    i、 bash(ksh)执行命令的过程:分析命令-变量求值-命令替代(``和$( ))-重定向-通配符展开-确定路径-执行命令;


    j、 ( ) 将 command group 置于 sub-shell 去执行,也称 nested sub-shell,它有一点非常重要的特性是:继承父shell的Standard input, output, and error plus any other open file descriptors。


    k、 exec 命令:常用来替代当前 shell 并重新启动一个 shell,换句话说,并没有启动子 shell。使用这一命令时任何现有环境都将会被清除。exec 在对文件描述符进行操作的时候,也只有在这时,exec 不会覆盖你当前的 shell 环境。


    2、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 时,这个就具有特殊的用途了。请大家务必理解这一点。


    3、 如果 stdin, stdout, stderr 进行了重定向或关闭, 但没有保存原来的 FD, 可以将其恢复到 default 状态吗?


    *** 如果关闭了stdin,因为会导致退出,那肯定不能恢复。


    *** 如果重定向或关闭 stdout和stderr其中之一,可以恢复,因为他们默认均是送往monitor(但不知会否有其他影响)。如恢复重定向或关闭的 stdout: exec 1>&2 ,恢复重定向或关闭的stderr:exec 2>&1。


    *** 如果stdout和stderr全部都关闭了,又没有保存原来的FD,可以用:exec 1>/dev/tty 恢复。


    4、 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将其打开。


    我想:他们的不同点在于:


    cmd >a 2>a 相当于使用了两个互相竞争使用文件a的管道;


    而cmd >a 2>&1 只使用了一个管道,但在其源头已经包括了stdout和stderr。


    从IO效率上来讲,cmd >a 2>&1的效率应该更高!


    exec 0exec 1>outfilename # 打开文件outfilename作为stdout


    exec 2>errfilename # 打开文件 errfilename作为 stderr


    exec 0&- # 关闭 FD1


    exec 5>&- # 关闭 FD5




    1.查询所有保护字符串“Hello”的文件


    find / -exec grep "Hello" {} \;


    2.删除所有临时文件


    find / -name "*.tmp" -exec rm -f {} \;


    3 删除20天以前的文件


    find / -name "*" -ctime +20 -exec rm -f {} \; (不要执行哦,要看清楚哈)
     

    展开全文
  • Linuxexec系列函数

    2019-10-06 21:32:33
    当进程调用exec系列函数中的任一个时,该进程用户空间资源(正文、数据、堆、栈)完全由新程序替代。因为调用exec并不创建新程序,如果无特殊指示代码,进程内核信息基本不用修改 (1)execl函数声明如下: // ...

    在这里插入图片描述
    当进程调用exec系列函数中的任一个时,该进程用户空间资源(正文、数据、堆、栈)完全由新程序替代。因为调用exec并不创建新程序,如果无特殊指示代码,进程内核信息基本不用修改

    (1)execl函数声明如下:

    // from /usr/include/unistd.h
    int execl(const char* path,const char* arg,...); 
    例子 : execl("/bin/ls","ls","-l","/home",(char*)0);
    

    第一个参数表示execl要执行的程序,第二个及以后的参数代表执行文件时传递的参数列表,最后一个参数必须是空指针。

    (2)execle函数声明如下:

    // from /usr/include/unistd.h
    int execle(const char* path,const char* arg,...); 
    例子:execle("/bin/ls","ls","-l","/home",(char*)0,env);
    

    execle用来执行参数path代表的程序,第二个及以后的参数代表执行文件时传入参数列表,最后一个参数必须指向一个新的环境变量数组,即新执行程序的环境变量。

    (3)execlp函数声明如下:

    // from /usr/include/unistd.h
    int execlp(const char* file,const char* arg,...); 
    例子:execlp("ls","ls","-l","/home",(char*)0);
    

    execlp会在$PATH环境变量所指向的目录中查找文件名为第一个参数的程序,并执行它,第二个及以后的参数代表执行文件时传递的参数列表,最后一个参数必须是空指针。

    (4)execv函数声明如下:

    // from /usr/include/unistd.h
    int execv(const char* path,const char* argv[]);
    例子:char* argv[] = {"ls","-l","/home",(char*)0};
    	 execv("/bin/ls",argv); 
    

    execv函数执行path路径下的程序,第二个参数为数组指针维护的程序参数列表,该数组的最后一个成员必须为NULL.

    (5)execvp函数声明如下:

    // from /usr/include/unistd.h
    int execvp(const char* file,const char* argv[]);
    例子:char* argv[] = {"ls","-l","/home",(char*)0};
    	 execv("ls",argv); 
    

    execvp会在$PATH环境变量所指向的目录中查找文件名为第一个参数的程序,并执行它,第二个及以后的参数代表执行文件时传递的参数列表,最后一个参数必须是空指针。

    (6)system函数声明如下:

    //from /usr/include/stdlib.h
    int system(const char* command);
    

    system函数用来创建新进程,直到程序结束后才继续运行父进程。一般运行脚本文件
    在这里插入图片描述

    展开全文
  • linux exec用法总结

    2015-01-14 12:02:52
    linux exec用法总结 先总结一个表: exec命令 作用 exec ls 在shell中执行ls,ls结束后不返回原来的shell中了 exec  将file中的内容作为exec的标准...

    linux exec用法总结

    先总结一个表:

    exec命令

    作用

    exec ls

    shell中执行lsls结束后不返回原来的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.shell script:

    有两种方法执行shell scripts,一种是新产生一个shell,然后执行相应的shell scripts;一种是在当前shell下执行,不再启用其他shell。

    新产生一个shell然后再执行scripts的方法是在scripts文件开头加入以下语句

    #!/bin/sh

    一般的script文件(.sh)即是这种用法。这种方法先启用新的sub-shell(新的子进程),然后在其下执行命令。

    另外一种方法就是上面说过的source命令,不再产生新的shell,而在当前shell下执行一切命令。


    2. 在bash下输入man exec,找到exec命令解释处,可以看到有”No new process is created.”这样的解释,这就是说exec命令不产生新的子进程。系统调用exec是以新的进程去代替原来的进程,但进程的PID保持不变。因此,可以这样认为,exec系统调用并没有创建新的进程,只是替换了原来进程上下文的内容。原进程的代码段,数据段,堆栈段被新的进程所代替。

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

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

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

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


    3. exec是一个函数簇,由6个函数组成,分别是以excl和execv打头的。

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


    4.exec I/O重定向详解及应用实例

    4.1 基本概念

    a、 I/O重定向通常与 FD有关,shell的FD通常为10个,即 0~9;

    b、 常用FD有3个,为0(stdin,标准输入)、1(stdout,标准输出)、2(stderr,标准错误输出),默认与keyboard、monitor、monitor有关;

    c、 用 > 来改变送出的数据信道(stdout, stderr),使之输出到指定的档案;

    e、 在IO重定向 中,stdout 与 stderr 的管道会先准备好,才会从 stdin 读进资料;

    f、 管道“|”(pipe line):上一个命令的 stdout 接到下一个命令的 stdin;

    g、 exec 命令:常用来替代当前 shell 并重新启动一个 shell,换句话说,并没有启动子 shell。使用这一命令时任何现有环境都将会被清除。exec 在对文件描述符进行操作的时候,也只有在这时,exec 不会覆盖你当前的 shell 环境。


    4.2、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 时,这个就具有特殊的用途了。请大家务必理解这一点。


    4.3、 如果 stdin, stdout, stderr 进行了重定向或关闭, 但没有保存原来的 FD, 可以将其恢复到 default 状态吗?

    *** 如果关闭了stdin,因为会导致退出,那肯定不能恢复。

    *** 如果重定向或关闭 stdout和stderr其中之一,可以恢复,因为他们默认均是送往monitor(但不知会否有其他影响)。如恢复重定向或关闭的 stdout: exec 1>&2 ,恢复重定向或关闭的stderr:exec 2>&1。

    *** 如果stdout和stderr全部都关闭了,又没有保存原来的FD,可以用:exec 1>/dev/tty 恢复。


    4.4、 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将其打开。

    我想:他们的不同点在于:

    cmd >a 2>a 相当于使用了两个互相竞争使用文件a的管道;

    而cmd >a 2>&1 只使用了一个管道,但在其源头已经包括了stdout和stderr。

    从IO效率上来讲,cmd >a 2>&1的效率应该更高!

    exec 0exec 1>outfilename # 打开文件outfilename作为stdout

    exec 2>errfilename # 打开文件 errfilename作为 stderr

    exec 0&- # 关闭 FD1

    exec 5>&- # 关闭 FD5


    展开全文
  • Linuxexec命令相关

    2012-02-12 22:15:17
    Linuxexec命令相关: exec和source都属于bash内部命令(builtins commands),在bash下输入man exec或man source可以查看所有的内部命令信息。  bash shell的命令分为两类:外部命令和内部命令。外部命令是通过...

    Linux中exec命令相关:

    exec和source都属于bash内部命令(builtins commands),在bash下输入man exec或man source可以查看所有的内部命令信息。
      bash shell的命令分为两类:外部命令和内部命令。外部命令是通过系统调用或独立的程序实现的,如sed、awk等等。内部

    命令是由特殊的文件格式(.def)所实现,如cd、history、exec等等。

      在说明exe和source的区别之前,先说明一下fork的概念。

      fork是linux的系统调用,用来创建子进程(child process)。子进程是父进程(parent process)的一个副本,从父进程那里

    获得一定的资源分配以及继承父进程的环境。子进程与父进程唯一不同的地方在于pid(process id)。

      环境变量(传给子进程的变量,遗传性是本地变量和环境变量的根本区别)只能单向从父进程传给子进程。不管子进程的环境

    变量如何变化,都不会影响父进程的环境变量。

      shell script:

      有两种方法执行shell scripts,一种是新产生一个shell,然后执行相应的shell scripts;一种是在当前shell下执行,不

    再启用其他shell。

      新产生一个shell然后再执行scripts的方法是在scripts文件开头加入以下语句

      #!/bin/sh

      一般的script文件(.sh)即是这种用法。这种方法先启用新的sub-shell(新的子进程),然后在其下执行命令。

      另外一种方法就是上面说过的source命令,不再产生新的shell,而在当前shell下执行一切命令。

      source:

      source命令即点(.)命令。

      在bash下输入man source,找到source命令解释处,可以看到解释”Read and execute commands from filename in the

    current shell environment and …”。从中可以知道,source命令是在当前进程中执行参数文件中的各个命令,而不是另起子

    进程(或sub-shell)。

      exec:

      在bash下输入man exec,找到exec命令解释处,可以看到有”No new process is created.”这样的解释,这就是说exec命

    令不产生新的子进程。那么exec与source的区别是什么呢?

      exec命令在执行时会把当前的shell process关闭,然后换到后面的命令继续执行。

     

    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、 用 来改变送出的数据信道(stdout, stderr),使之输出到指定的档案;

      e、 0 是 与 1> 是一样的;

      f、 在IO重定向 中,stdout 与 stderr 的管道会先准备好,才会从 stdin 读进资料;

      g、 管道“|”(pipe line):上一个命令的 stdout 接到下一个命令的 stdin;

      h、 tee 命令是在不影响原本 I/O 的情况下,将 stdout 复制一份到档案去;

      i、 bash(ksh)执行命令的过程:分析命令-变量求值-命令替代(``和$( ))-重定向-通配符展开-确定路径-执行命令;

      j、 ( ) 将 command group 置于 sub-shell 去执行,也称 nested sub-shell,它有一点非常重要的特性是:继承父shell

    的Standard input, output, and error plus any other open file descriptors。

      k、 exec 命令:常用来替代当前 shell 并重新启动一个 shell,换句话说,并没有启动子 shell。使用这一命令时任何现

    有环境都将会被清除。exec 在对文件描述符进行操作的时候,也只有在这时,exec 不会覆盖你当前的 shell 环境。

      2、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 时,这个就具有特殊的用途了。请大家务必理解这一点。

      3、 如果 stdin, stdout, stderr 进行了重定向或关闭, 但没有保存原来的 FD, 可以将其恢复到 default 状态吗?

      *** 如果关闭了stdin,因为会导致退出,那肯定不能恢复。

      *** 如果重定向或关闭 stdout和stderr其中之一,可以恢复,因为他们默认均是送往monitor(但不知会否有其他影响)。如

    恢复重定向或关闭的 stdout: exec 1>&2 ,恢复重定向或关闭的stderr:exec 2>&1。

      *** 如果stdout和stderr全部都关闭了,又没有保存原来的FD,可以用:exec 1>/dev/tty 恢复。

      4、 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

    将其打开。

      我想:他们的不同点在于:

      cmd >a 2>a 相当于使用了两个互相竞争使用文件a的管道;

      而cmd >a 2>&1 只使用了一个管道,但在其源头已经包括了stdout和stderr。

      从IO效率上来讲,cmd >a 2>&1的效率应该更高!

      exec 0exec 1>outfilename # 打开文件outfilename作为stdout

      exec 2>errfilename # 打开文件 errfilename作为 stderr

      exec 0&- # 关闭 FD1

      exec 5>&- # 关闭 FD5

    [linuxidc.com@linuxidc.com shell]$ cat 1

      11 22 33 44 55

      66 22 33 11 33

      324 25 63 634 745

      [linuxidc.com@linuxidc.com shell]$ cat 2

      > 1.txt

      exec 4<&1

      exec 1> 1.txt

      while read line

      do

      echo $line

      done < 1

      exec 1<&4

      exec 4>&-

      [linuxidc.com@linuxidc.com shell]$ sh ./2

      [linuxidc.com@linuxidc.com shell]$ cat 1.txt

      11 22 33 44 55

      66 22 33 11 33

      324 25 63 634 745

      [linuxidc.com@linuxidc.com shell]$

     

    修改ls显示的时间格式:

    ls -l --time-style '+%Y/%m/%d %H:%M:%S'
    total 0
    -rw-r--r-- 1 root root 0 2008/08/01 12:23:06 file1
    -rw-r--r-- 1 root root 0 2008/08/01 12:23:06 file2
     
    --time-style 也可以存在环境变量里:
     
    export TIME_STYLE='+%Y/%m/%d %H:%M:%S'

    ls -l
    total 0
    -rw-r--r-- 1 root root 0 2008/08/01 12:23:06 file1
    -rw-r--r-- 1 root root 0 2008/08/01 12:23:06 file2

     

     

    Linux中内部命令和外部命令:
       UNIX命令有内部命令和外部命令之分。内部命令实际上是shell程序的一部分,其中包含的是一些比较简练的UNIX系统命令,这些命令由shell程序识别并在shell程序内部完成运行,通常在UNIX系统加载运行时shell就被加载并驻留在系统内存中。外部命令是UNIX系统中的实用程序部分,因为实用程序的功能通常都比较强大,所以它们包含的程序量也会很大,在系统加载时并不随系统一起被加载到内存中,而是在需要时才将其调进内存。通常外部命令的实体并不包含在shell中,但是其命令执行过程是由shell 程序控制的。shell程序管理外部命令执行的路径查找、加载存放,并控制命令的执行。
    理解了内部命令和外部命令之后,你可能就会有疑问了??那么我怎么看一个命令是内部命令还是外部命令呢??

    1、在终端下输入man builtins,即可调出内部命令的大纲和解释。

    2、用命令enable调出来,后者将分行打印出内部命令

         [root@localhost shell]# enable ls

          bash: enable: ls: not a shell builtin


    1. http://blog.sina.com.cn/s/blog_6238358c0100sg5n.html

    展开全文
  • linux exec 和 重定向

    2018-04-03 14:19:58
    Linuxexec命令相关:exec和source都属于bash内部命令(builtins commands),在bash下输入man exec或man source可以查看所有的内部命令信息。 bash shell的命令分为两类:外部命令和内部命令。外部命令是通过系统...

    Linux中exec命令相关:

    exec和source都属于bash内部命令(builtins commands),在bash下输入man exec或man source可以查看所有的内部命令信息。
      bash shell的命令分为两类:外部命令和内部命令。外部命令是通过系统调用或独立的程序实现的,如sed、awk等等。内部

    命令是由特殊的文件格式(.def)所实现,如cd、history、exec等等。

      在说明exe和source的区别之前,先说明一下fork的概念。

      fork是linux的系统调用,用来创建子进程(child process)。子进程是父进程(parent process)的一个副本,从父进程那里

    获得一定的资源分配以及继承父进程的环境。子进程与父进程唯一不同的地方在于pid(process id)。

      环境变量(传给子进程的变量,遗传性是本地变量和环境变量的根本区别)只能单向从父进程传给子进程。不管子进程的环境

    变量如何变化,都不会影响父进程的环境变量。

      shell script:

      有两种方法执行shell scripts,一种是新产生一个shell,然后执行相应的shell scripts;一种是在当前shell下执行,不

    再启用其他shell。

      新产生一个shell然后再执行scripts的方法是在scripts文件开头加入以下语句

      #!/bin/sh

      一般的script文件(.sh)即是这种用法。这种方法先启用新的sub-shell(新的子进程),然后在其下执行命令。

      另外一种方法就是上面说过的source命令,不再产生新的shell,而在当前shell下执行一切命令。

      source:

      source命令即点(.)命令。

      在bash下输入man source,找到source命令解释处,可以看到解释”Read and execute commands from filename in the

    current shell environment and …”。从中可以知道,source命令是在当前进程中执行参数文件中的各个命令,而不是另起子

    进程(或sub-shell)。

      exec:

      在bash下输入man exec,找到exec命令解释处,可以看到有”No new process is created.”这样的解释,这就是说exec命

    令不产生新的子进程。那么exec与source的区别是什么呢?

      exec命令在执行时会把当前的shell process关闭,然后换到后面的命令继续执行。

     

    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、 用 来改变送出的数据信道(stdout, stderr),使之输出到指定的档案;

      e、 0 是 与 1> 是一样的;

      f、 在IO重定向 中,stdout 与 stderr 的管道会先准备好,才会从 stdin 读进资料;

      g、 管道“|”(pipe line):上一个命令的 stdout 接到下一个命令的 stdin;

      h、 tee 命令是在不影响原本 I/O 的情况下,将 stdout 复制一份到档案去;

      i、 bash(ksh)执行命令的过程:分析命令-变量求值-命令替代(``和$( ))-重定向-通配符展开-确定路径-执行命令;

      j、 ( ) 将 command group 置于 sub-shell 去执行,也称 nested sub-shell,它有一点非常重要的特性是:继承父shell

    的Standard input, output, and error plus any other open file descriptors。

      k、 exec 命令:常用来替代当前 shell 并重新启动一个 shell,换句话说,并没有启动子 shell。使用这一命令时任何现

    有环境都将会被清除。exec 在对文件描述符进行操作的时候,也只有在这时,exec 不会覆盖你当前的 shell 环境。

      2、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 时,这个就具有特殊的用途了。请大家务必理解这一点。

      3、 如果 stdin, stdout, stderr 进行了重定向或关闭, 但没有保存原来的 FD, 可以将其恢复到 default 状态吗?

      *** 如果关闭了stdin,因为会导致退出,那肯定不能恢复。

      *** 如果重定向或关闭 stdout和stderr其中之一,可以恢复,因为他们默认均是送往monitor(但不知会否有其他影响)。如

    恢复重定向或关闭的 stdout: exec 1>&2 ,恢复重定向或关闭的stderr:exec 2>&1。

      *** 如果stdout和stderr全部都关闭了,又没有保存原来的FD,可以用:exec 1>/dev/tty 恢复。

      4、 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

    将其打开。

      我想:他们的不同点在于:

      cmd >a 2>a 相当于使用了两个互相竞争使用文件a的管道;

      而cmd >a 2>&1 只使用了一个管道,但在其源头已经包括了stdout和stderr。

      从IO效率上来讲,cmd >a 2>&1的效率应该更高!

      exec 0exec 1>outfilename # 打开文件outfilename作为stdout

      exec 2>errfilename # 打开文件 errfilename作为 stderr

      exec 0&- # 关闭 FD1

      exec 5>&- # 关闭 FD5

    [linuxidc.com@linuxidc.com shell]$ cat 1

      11 22 33 44 55

      66 22 33 11 33

      324 25 63 634 745

      [linuxidc.com@linuxidc.com shell]$ cat 2

      > 1.txt

      exec 4<&1

      exec 1> 1.txt

      while read line

      do

      echo $line

      done < 1

      exec 1<&4

      exec 4>&-

      [linuxidc.com@linuxidc.com shell]$ sh ./2

      [linuxidc.com@linuxidc.com shell]$ cat 1.txt

      11 22 33 44 55

      66 22 33 11 33

      324 25 63 634 745

      [linuxidc.com@linuxidc.com shell]$

     

    修改ls显示的时间格式:

    ls -l --time-style '+%Y/%m/%d %H:%M:%S'
    total 0
    -rw-r--r-- 1 root root 0 2008/08/01 12:23:06 file1
    -rw-r--r-- 1 root root 0 2008/08/01 12:23:06 file2
     
    --time-style 也可以存在环境变量里:
     
    export TIME_STYLE='+%Y/%m/%d %H:%M:%S'

    ls -l
    total 0
    -rw-r--r-- 1 root root 0 2008/08/01 12:23:06 file1
    -rw-r--r-- 1 root root 0 2008/08/01 12:23:06 file2

     

     

    Linux中内部命令和外部命令:
       UNIX命令有内部命令和外部命令之分。内部命令实际上是shell程序的一部分,其中包含的是一些比较简练的UNIX系统命令,这些命令由shell程序识别并在shell程序内部完成运行,通常在UNIX系统加载运行时shell就被加载并驻留在系统内存中。外部命令是UNIX系统中的实用程序部分,因为实用程序的功能通常都比较强大,所以它们包含的程序量也会很大,在系统加载时并不随系统一起被加载到内存中,而是在需要时才将其调进内存。通常外部命令的实体并不包含在shell中,但是其命令执行过程是由shell 程序控制的。shell程序管理外部命令执行的路径查找、加载存放,并控制命令的执行。 
    理解了内部命令和外部命令之后,你可能就会有疑问了??那么我怎么看一个命令是内部命令还是外部命令呢??

    1、在终端下输入man builtins,即可调出内部命令的大纲和解释。

    2、用命令enable调出来,后者将分行打印出内部命令

         [root@localhost shell]# enable ls

          bash: enable: ls: not a shell builtin



    出处:http://blog.sina.com.cn/s/blog_6238358c0100sg5n.html

    展开全文
  • linux--exec命令

    2018-07-04 20:18:50
    转载的文章:https://blog.csdn.net/clozxy/article/details/5818465https://blog.csdn.net/sunnyliqian/article/details/50244273shell的内建命令exec将并不启动新的shell,而是用要被执行命令替换当前的shell进程...
  • linux find之exec用法

    2018-10-12 16:46:02
    find命令之exec -exec参数后面跟的是command命令,它是以 ; 为结束标志,由于各个系统中分号会有不同的意义,因此在前面加上反斜杠。 {} 代表前面find查找出来的文件名 [danni@vm-xxx-18 study]$ find ./ -type f -...
  • Linuxexec命令相关:

    2018-09-12 15:15:49
    exec和source都属于bash内部命令(builtins commands),在bash下输入man exec或man source可以查看所有的内部命令信息。  bash shell的命令分为两类:外部命令和内部命令。外部命令是通过系统调用或独立的程序实现的...
  • Linuxexec与xargs都是将前一个命令的结果作为他的标准输入,但二者还是有区别的。 -exec: 要结合{};来用, {}表示前面查询的结果;结束符,这是固定的写法 参数是一个一个传递的,传递一个参数执行一次 文件名有...
  • 转自: ... 使用sh -c , 不然会报 can not create Process error =2的错误! 一、Runtime.getRuntime().exec()执行命令情况:  ... Runtime.getRuntime().exec("cmd /c your command")
  • Exec函数: 当使用fork函数创建子进程后,子进程往往会调用一种exec函数以执行另一个程序,当进程调用一种exec函数时,该进程执行的程序完全被替换为新的程序,而新的程序则从其main函数开始执行,因为调用exec并不...
  • 近期在使用phantomjs做图片导出功能时发现一个奇怪的问题,java端使用Runtime.getRuntime().exec(cmd)调用一条语句,windows下执行正常,linux下报错。样例语句如下: /home test.sh "2018-06-06 11:00" ...
  • linux exec的用法

    2013-02-28 10:18:09
    说是exec系统调用,实际上在Linux中,并不存在一个exec()的函数形式,exec指的是一组函数,一共有6个,分别是: #include extern char **environ; int execl(const char *path, const ch
  • Linux命令--find命令之execexec解释:实例1:ls -l 命令放在 find 命令的 -exec 选项中实例2:在目录中查找更改时间在n 日以前的文件并删除它们实例3:在目录中查找更改时间在n日以前的文件并删除它们,在删除之前...
  • (1)exec函数说明 fork函数是用于创建一个子进程,该子进程几乎是父进程的副本,而有时我们希望子进程去执行另外的程序,exec函数族就提供了一个在进程中启动另一个程序执行的方法。它可以根据指定的文件名或目录...
  • type=note 目录 一、exec族函数及实战 1、exec族函数包含如下函数: 2、函数使用示例: 二、进程状态和system函数 1、进程的5种状态 2、进程各种状态之间的转换图 3、system...
  • Linux C编程中,调用另一个可执行文件或调用命令可以使用system函数和exec系列的函数。 下面关于exec系列函数的一些方法和使用做一个简单的记录。 exec系列函数。都是以execl开头或者execv开头(l表示列表list,v...
  • Linux exec简介

    2017-06-12 22:13:54
    exec函数简介
1 2 3 4 5 ... 20
收藏数 199,902
精华内容 79,960
关键字:

9 exec linux