Linux标准输入、重定向与参数传递

按惯例,每当运行一个新程序时,所有shell都为其打开3个文件描述符,即标准输入、标准输出以及标准错误。如果不做特殊处理,例如就像简单的命令ls,则这三个描述符都链接向终端。大多数shell都提供一种方法,使其中一个或所有这3个描述符都能重新定向到某个文件。

——《UNIX环境高级编程(第三版)》

也就是说,我们可以用文件内容来替代从终端输入的内容,也可以用文件来代替终端接收程序输出的内容。于是,我写了个简单的求两数和的小程序测试重定向功能:

#include <stdio.h>
int main(int argc,char** argv)
{
int a,b;
a=argv[1][0]-'0';
b=argv[2][0]-'0';
printf("The sum is %d\n.",a+b);
exit(0);
}

编译,测试:

gcc add.c -o add
./add 1 2

输出:

The sum is 3.

没有问题。

继续测试重定向,测试文件input的内容只有两个数字:

1 2

输入命令:

./add < input

得到的输出却是:

Segmentation fault (core dumped)

奇怪,为什么重定向不起作用了,文件中的内容和我在命令行中输入的内容完全一样,为什么却引起了错误?

然后我想明白了,标准输入、标准输出、标准错误对于程序来说,都是文件,它们确实默认被链接到终端上了。但是这并不是说从终端输入的所有内容都属于标准输入,当我们输入命令

./add 1 2

时,这条语句不属于标准输入,而是作为一条“命令”传给命令解释器的,而其后跟随的两个数字“1 2”自然也不属于标准输入,而是属于“命令”的一部分——作为参数一起传递给命令解释器(Command Interpreter)的。

同时,源文件中的main函数括号中的 argc 和 argv 所接收的也不是标准输入的内容,而是由命令解释器传递过来的参数"1" "2"。整个程序压根就没有从标准输入获取数据,只是接收了命令解释器传过来的“参数”,做的运算也是基于“参数”的。

基于这两点,之前的错误就好理解了,当我们输入:

./add < input

时,程序 add 对应的标准输入的确被重定向为了文件 input,但是由于它不从标准输入读内容,所以这个重定向对它来说没有意义。又由于这条语句中没有向add传递其程序中所需要的参数,在运行过程中当需要用到两个参数时就会找不到,于是发生"Segmentation fault" 的错误。

意识到这一点后,重写一下源代码,把两个加数的来源改一下:

#include <stdio.h>
int main(void)
{
int a,b;
scanf("%d %d",&a,&b); //这才是从标准输入中获取数据
printf("The sum is %d.\n",a+b);
exit(0);
}

编译:

gcc add.c -o add

运行,并输入两个加数、得到结果:

./add
1 2
The sum is 3.

再测试重定向:

./add < input
The sum is 3.

这样一来就OK了。

有个问题:为什么其他程序就可以通过重定向来正常运行?比如:

cat input
cat < input

输出都是input文件的内容:

1 2

为什么cat就可以同时适用这两个方法?这两条命令其实机制不一样,第一条是cat程序接收命令解释器传来的参数字符串"input",然后根据这个字符串参数在目录中搜索到名为"input"的文件,再获取内容,进而运行。而第二条则是通过重定向把cat的标准输入替换成input文件,cat直接从这个文件中获取其内容,进而运行。因此,虽然这两条命令使用的是相同的程序、读取的是相同的文件、输出的是同样的结果,但程序获取文件内容的途径是完全不一样的。

总结:

  • 标准输入、标准输出、标准错误都被程序当做文件,且自动链接到终端上。

  • 但并非所有从终端输入的内容都属于标准输入,有时只是作为命令的参数来传递给程序的。Linux下程序获取内容有两种方式:一种是从文件输入(从终端或控制台手动输入也属于文件),一种是在程序启动的同时就从命令解释器直接传递(注意,叫“传递”而不是叫“输入”)过来的命令参数,它们有时可以完成相同的效果,但这完全是两种机理。

  • 那么,从终端输入的内容,什么时候是“标准输入”、什么时候是“参数传递”呢?

    • 答:伴随程序启动时就附在命令后面的内容属于参数,当程序启动之后才从终端输入的才叫“标准输入”(标准输入本质上就是一种文件)。
    • 因为,所谓的“标准输入”、“标准输出”、“标准错误”等文件描述符,都是相对于程序而言的,只有当程序启动之后它们才被建立。也就是说,虽然都叫“标准输入/输出/错误”、虽然都是从终端输入/输出,但不同程序的“标准输入”是不一样的,只不过都被默认链接到终端上了而已。
    • 比如,假设你在前台运行程序A,当你在终端上把程序A调到后台运行,把程序B调到前台时,你从终端上输入的内容已经不再属于A而是属于B了。这是因为你把A调到后台之后,A的标准输入就与终端断开了联系,而B的标准输入开始与终端建立起联系来,于是,你的输入的内容自然不归A而归B了。

码字不易,倘若觉得分享有益,就点个赞再走呗~

Linux标准输入、重定向与参数传递的更多相关文章

  1. Linux标准输入、输出和错误和文件重定向(转) --- good

    标准输入.输出和错误 当我们在shell中执行命令的时候,每个进程都和三个打开的文件相联系,并使用文件描述符来引用这些文件.由于文件描述符不容易记忆,shell同时也给出了相应的文件名.下面就是这些文 ...

  2. Linux标准输入、输出和错误和文件重定向 专题

    当我们在shell中执行命令的时候,每个进程都和三个打开的文件相联系,并使用文件描述符来引用这些文件.由于文件描述符不容易记忆,shell同时也给出了相应的文件名. 下面就是这些文件描述符及它们通常所 ...

  3. linux --stdin 管道 标准输入重定向

    linux --stdin 标准输入重定向 --stdin This option is used to indicate that passwd should read the new passwo ...

  4. Linux Shell 重定向与管道【转帖】

    by 程默 在了解重定向之前,我们先来看看linux 的文件描述符. linux文件描述符:可以理解为linux跟踪打开文件,而分配的一个数字,这个数字有点类似c语言操作文件时候的句柄,通过句柄就可以 ...

  5. 全面理解Linux输入输出重定向

    全面理解Linux输入输出重定向 本教程通过视频方式讲解shell操作,理解感念,教程通俗易懂,比起看一大堆文档要舒服的多.本次教程主要讲解  Linux Shell 中支持输入输出重定向,用符号&l ...

  6. 十句话教你学会Linux数据流重定向

    1.看到重定向一下子就想起了web里面的redirect,没错,但是Linux数据流重定向的作用不是跳到另一个网页,而是用来存储重要的屏幕信息.将不必要的屏幕信息输出到文件里或者“黑洞”里.将错误信息 ...

  7. Linux数据流重定向与管道

    数据流重定向简单来说就是把原本应该输出到某处(比如说屏幕)的数据,重定向其输出目的地,到其他的地方(比如文件). linux中的输入与输出: 标准输入(stdin):默认从键盘输入 标准输出(stdo ...

  8. Linux shell 重定向学习笔记

    在了解重定向之前,我们先来看看linux 的文件描述符. linux文件描述符:可以理解为linux跟踪打开文件,而分配的一个数字,这个数字有点类似c语言操作文件时候的句柄,通过句柄就可以实现文件的读 ...

  9. linux命令重定向>、>>、 1>、 2>、 1>>、 2>>、 <

    重定向命令其实用得不少吧,只是重来都没有仔细看过,这波正好又用到 又有空总结一波. 先看>和>>: 他们俩其实唯一的区别就是>是重定向到一个文件,>>是追加内容到文 ...

随机推荐

  1. CSS3 & CSS var & :root

    CSS3 & CSS var & :root How to change CSS :root color variables in JavaScript https://stackov ...

  2. mdn & remove & removeChild

    mdn & remove & removeChild Element https://developer.mozilla.org/en-US/docs/Web/API/Element ...

  3. NGK的发行量是多少?NGK销毁机制是怎么样的?

    代币销毁(Coin Burning),是指将代币从流通中永久性去除.换句话说,被销毁的代币相当于被永久性冻结,再也无法流入市场.那为什么要进行代币销毁呢? 销毁加密货币,可以使剩余加密货币的价值升高, ...

  4. BGV劝早买内存

    12月3日,BGV全球首发,上线AOFEX交易所(A网),全球区块链爱好者震惊.很多人争相抢挖BGV,希望能够及早获取BGV带来的红利.有趣的是,随着BGV抢挖人数的增多,NGK内存也迎来了暴涨,在1 ...

  5. python中的enumerate 函数(编号的实现方式)

    enumerate 函数用于遍历序列中的元素以及它们的下标: 默认从0开始,如果想从1开始,可以仿照最后案例 加上逗号,和数字编号 >>> for i,j in enumerate( ...

  6. 渗透测试--Nmap主机识别

    通过本篇博客可以学到:Nmap的安装和使用,列举远程机器服务端口,识别目标机器上的服务,指纹,发现局域网中存活主机,端口探测技巧,NSE脚本使用,使用特定网卡进行检测,对比扫描结果ndiff,可视化N ...

  7. Python分类模型构建

    分离训练集测试集 from sklearn.model_selection import train_test_split eg: X_train, X_test, y_train, y_test = ...

  8. apiAutoTest:支持自定义函数,用例中可调用

    0. 前言 apiAutoTest从去年8月以来开源至今,也更新了不少内容,一起来看看吧 第一个版本 - 2020/08/08 增加实际响应存储数据的方法,并在字典可以处理依赖见tools/svae_ ...

  9. 基于solarflare的openonload技术以TCPDirect方法加速epoll

    [前言]基于solarflare的onload模式加速,官方文档给出TCPDirect模式可以实现从300ns到30ns的延迟缩减.我们需要测试在我们的交易模型框架中他的延时,有人给出了tcpdire ...

  10. JavaScript实现动态添加员工

    html代码: <div id="empAdd"> <fieldset> <legend><strong>添加员工</stro ...