看到好几篇文章讲述exec都是一知半解,所以我尽量说的清楚明白一些。本文首先讲述Linux文件描述符,然后是exec,最后举例说明exec I/O重定向及其用法。

概念:exec命令用于调用并执行指令的命令。exec命令通常用在shell脚本程序中,可以调用其他的命令。如果在当前终端中使用命令,则当指定的命令执行完毕后会立即退出终端。
语法:
exec(选项)(参数)
选项:
-c:在空环境中执行指定的命令。
参数:
指令:要执行的指令和相应的参数。
实例:
首先使用echo命令将文本“Linux C++”进行输出,输入如下命令:
echo Linux C++ #输出指定信息
执行上面的指令后,输出如下信息:
Linux C++ #输出信息
然后再使用exec命令调用echo命令输出同样的信息,并且对输出的信息进行对比,输入指令如下所示:
exec -c echo Linux C++ #调用命令
执行以上命令后,其输出信息如下(并退出当前终端):
Linux C++ #使用指定指令输出信息 通过比较两者执行后的结果来看,所实现的功能是相同的,即使用exec命令调用echo命令成功。

1 Linux文件描述符介绍

在Linux系统中一切皆可以看成是文件,文件又可分为:普通文件、目录文件、链接文件和设备文件。文件描述符(file descriptor简称FD)是内核为了高效管理已被打开的文件所创建的索引,其是一个非负整数(通常是小整数),用于指代被打开的文件,所有执行I/O操作的系统调用都通过文件描述符。最前面的三个文件描述符(0,1,2)分别与标准输入(stdin),标准输出(stdout)和标准错误(stderr)对应。因此,函数 scanf() 使用 stdin,而函数 printf() 使用 stdout。你可以用不同的文件描述符改写默认的设置并重定向进程的 I/O 到不同的文件。Unix 操作系统通常给每个进程能打开的文件数量强加一个限制。

例如:要把标准输出(1)和标准错误(2)重定向到一个文件,使用如下命令

xx命令 >& filename

查看LINUX默认的文件描述符数:

# ulimit -n

使用用如下命令来增大文件描述符数:

ulimit -HSn 

2 exec原理介绍

exec和source都属于bash内部命令(builtins commands),
    source:

source命令即点(.)命令。

在bash下输入man source,找到source命令解释处,可以看到解释”Read and execute commands from filename in thecurrent shell environment and …”。从中可以知道,source命令是在当前进程中执行参数文件中的各个命令,而不是另起子进程(或sub-shell)。

exec:

在bash下输入man exec,找到exec命令解释处,可以看到有”No new process is created.”这样的解释,这就是说exec命令不产生新的子进程。那么exec与source的区别是什么呢? exec命令在执行时会把当前的shell process关闭,然后换到后面的命令继续执行。

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

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

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

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

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

描述exec命令最贴切的说法是:它践踏了你当前的shell。

当这个脚本结束了,相应的会话可能也就结束了。

但是,exec在对文件描述符进行操作的时候(也只有在这个时候),它不会覆盖你当前的shell。

3 exec I/O重定向基本概念和应用案例

3.1 基本概念(可以大体看一下意思,等看到后面的例子再来参考)

1 常用FD(文件描述符)有3个,为0(stdin,标准输入)、1(stdout,标准输出)、2(stderr,标准错误输出),但你也可以指定其他数字作为文件描述符;

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

exec用法总结

exec命令

作用

exec ls

在shell中执行ls,ls结束后不返回原来的shell中了

exec <file

将file中的内容作为exec的标准输入

exec >file

将file中的内容作为标准写出

exec 3<file

将file读入到fd3中

exec 3>file

将写入fd3中的内容写入file中

sort <&3

fd3中读入的内容被分类

ls >&4

Ls将不会有显示,直接写入fd4中了,即上面的file中,此处ls也可以替换为任何命令

exec 5<&4

创建fd4的拷贝fd5

exec 3<&-

关闭fd3

1、通过exec分配文件描述符3.2 应用案例
#!/bin/sh
exec <>hello.txt           # 以读写方式绑定到文件描述符"
       # 写入"hello exec",如果之前有内容,这里将会从文件开头进行覆盖
      # 写入"hello world“,新的一行!
exec >&-                   # 关闭写,禁止写,然而,实际上它也不能读了~
                            # 如果是exec <&-,关闭读,同时它也不能写了~

打开文件,看到的结果如下

[root@vmtest01 tmp]# cat hello.txt
hello exec                                 #首次插入的在第一行
hello world        #第二次在新的一行
errfilename        #下面的都是旧数据,依旧保留
hello.txt
systemd-private-CEGg3c
test

2、将标准输出重定向到hello.txt

#!/bin/sh
exec >hello.txt        # 将标准输出重定向到文件hello.txt,从此以后,当前环境(shell)中的的标准输出都将被写入文件hello.txt
echo "hello exec"
echo "hello world"

这样,在终端再也见不着标准输出了,因为所有标准输出到放大文件hello.txt中了
如果你现在运行who am i命令,终端什么都不显示,打开另外一个终端

可以发现hello.txt文件的内容如下

[root@vmtest01 tmp]# cat hello.txt
hello exec
hello world
root     pts/        -- : (192.168.233.1)  #这是who am i命令的输出,已经重定向到该文件中了

3、在上面这个示例中,标准输出被重定向了,如果要恢复怎么办?

#!/bin/sh
exec >&             # 将文件描述符100连接到标准输出,此时100 和1同时指向hello.txt,即echo  和echo "hello exec"的效果一样,都写入hello.txt
                        # 由于之后还要输出到终端,所以我们不得不使用一个临时的描述符来保存它!
exec >hello.txt        # 首先清空hello.txt,并再次将标准输出重定向到文件hello.txt,从此以后,这个环境中的标准输出都将被写入文件hello.txt
echo "hello exec"
exec >& >&-      # 将标准输出连接到100,这是之前保存的标准输出
                        # 将描述符100关了,一了百了,因为已经还原标准输出了,留着它实在没必要
echo "oh, my god!"      # 从这句开始将显示在终端上

4、标准输入的操作与2、3相似,只是使用的是"<"

注意,下面这个必须存到脚本里面执行,不然总是会退出登录

#!/bin/sh
exec <&
exec <hello.txt      #假设hello.txt只有一行
read line1
echo $line1
exec <& >&-     # 将标准输入连接到100
                        # 将描述符100关了,一了百了,因为已经还原标准输入了,留着它实在没必要
read -p "please input:"    custome    #测试不是文件的标准输入
echo "your input is :$custome"

参考

linux shell的一些技巧(五)exec与文件描述符

Linux中exec命令相关

linux exec和文件描述符妙用技巧

man exec命令

linux exec用法总结 比较好的讲解

Linux exec与文件描述符的更多相关文章

  1. linux exec和文件描述符妙用技巧(转)

    最近在看<精通unix shell脚本编程>时,看到exec<$1 exec 1>$OUTFILE,一下看的我就蒙了.网上看了大半天,终于搞定,记录如下.对于 Linux 而言 ...

  2. linux exec操作文件描述符

    linux每一个打开文件都会关联一个文件描述符,需要的时候我们可以使用exec命令指定一个大于3的数字作为文件 linux默认文件描述符 每打开一个shell就会打开默认的三个文件描述符描0,1,2, ...

  3. Linux中的文件描述符与打开文件之间的关系

    Linux中的文件描述符与打开文件之间的关系 导读 内核(kernel)利用文件描述符(file descriptor)来访问文件.文件描述符是非负整数.打开现存文件或新建文件时,内核会返回一个文件描 ...

  4. Linux Linux下最大文件描述符设置

    Linux下最大文件描述符设置 by:授客 QQ:1033553122 1.   系统可打开最大文件描述符设置 查看系统可打开最大文件描述符 # cat /proc/sys/fs/file-max 6 ...

  5. [svc]linux中的文件描述符(file descriptor)和文件

    linux中的文件描述符(file descriptor)和文件 linux为了实现一切皆文件的设计哲学,不仅将数据抽象成了文件,也将一切操作和资源抽象成了文件,比如说硬件设备,socket,磁盘,进 ...

  6. linux fcntl 对文件描述符控制

    linux fcntl 对文件描述符控制 linux fcntl 对文件描述符控制 linux fcntl 对文件描述符控制

  7. linux shell exec 关联文件描述符

    在写shell脚本时,如果多个命令的输入或输出都是同一个文件,而这个文件的路径和名字都很长,则需要书写很多次同样的路径会很浪费时间,我们可以使用exec命令来关联一个自定义的文件描述符到一个特定的文件 ...

  8. Linux中的文件描述符与打开文件之间的关系------------每天进步一点点系列

    http://blog.csdn.net/cywosp/article/details/38965239 1. 概述     在Linux系统中一切皆可以看成是文件,文件又可分为:普通文件.目录文件. ...

  9. (转)Linux中的文件描述符

    本文转自:http://blog.csdn.net/cywosp/article/details/38965239 作者:cywosp 1. 概述 在Linux系统中一切皆可以看成是文件,文件又可分为 ...

随机推荐

  1. Mysql无法启动 InnoDB: Attempted to open a previously opened tablespace

    win2008,Mysql5.6,mysql服务无法启动 查看事件日志,报错InnoDB: Attempted to open a previously opened tablespace 最终解决方 ...

  2. Android Studio C/C++开发环境配置

    Android Studio C/C++开发环境配置  我的开发环境 : Win 10 + android studio 1.5   一, 安装NDK 开发环境: 1.  Settings -> ...

  3. 分享用于学习C++音频处理的代码示例

    与<分享用于学习C++图像处理的代码示例>为姊妹篇. 为了便于学习C++音频处理并研究音频算法, 俺写了一个适合初学者学习的小小框架. 麻雀虽小五脏俱全,仅仅考虑单通道处理. 采用Deco ...

  4. 数据结构-栈(应用篇)之快速排序法-C和C++的实现

    一.原理解析 快速排序法: 基本思路是,从第一个元素开始,把所有比它大的元素放在它后面,把所有比它小的元素放前面.然后划分它前面和后面的所有元素,分别再做快速排序,直到无法再划分为止. 在以下程序案例 ...

  5. ES6小点心第二弹——底部浮现弹窗

    小点心,顾名思义,开箱即食,拿来即用. 献上第二个小点心:SlidePopup. GitHub 在线演示 GitHub 上欢迎大家来找茬^_^ 前端朋友们,今天要介绍的这款小点心牛B了.相信是个前端都 ...

  6. Python 集合 深浅copy

    一,集合. 集合是无序的,不重复的数据集合,它里面的元素是可哈希的(不可变类型),但是集合本身是不可哈希(所以集合做不了字典的键)的.以下是集合最重要的两点: 去重,把一个列表变成集合,就自动去重了. ...

  7. Elixir的Phoenix框架:请求处理之道

    本文基于Phoenix1.3,但请求的处理流程跟1.2基本一致,只是模块的命名和目录结构有所差异. 简单介绍,phoenix是一个网站框架,本质就是http请求处理.这篇文章主要就是讲一个请求,在结果 ...

  8. LibreOJ β Round #2 F. 数学上来先打表

    传送门 题解 做法与题解基本无异,不过他说用vector我觉得用链表来得更好一些. #include<queue> #include<ctime> #include<bi ...

  9. 最小生成树—prim算法

    最小生成树prim算法实现 所谓生成树,就是n个点之间连成n-1条边的图形.而最小生成树,就是权值(两点间直线的值)之和的最小值. 首先,要用二维数组记录点和权值.如上图所示无向图: int map[ ...

  10. 对于hive使用的一点记录

    最近一段时间因工作需要接触了一些hive上的使用!当然大部分都是比较基本的使用,仅当入门!各位看到有不足之处望多多指正! 废话不多说,开始: 首先是创建数据库 create database '数据库 ...