[Linux] 文件描述符

来源:http://blog.csdn.net/xlinsist/article/details/51147212

0x00 Linux标准文件描述符

文件描述符 缩写 描述
0 STDIN 标准输入
1 STDOUT 标准输出
2 STDERR 标准错误输出

Linux系统将所有设备都当作文件来处理,而Linux用文件描述符来标识每个文件对象。其实我们可以想象我们电脑的显示器和键盘在Linux系统中都被看作是文件,而它们都有相应的文件描述符与之对应。

其实我们与计算机之间的交互是我可以输入一些指令之后它给我一些输出。
那么我们可以把上面表格中的文件描述符0理解为我和计算机交互时的输入,而这个输入默认是指向键盘的;
文件描述符1理解为我和计算机交互时的输出,而这个输出默认是指向显示器的;
文件描述符2理解为我和计算机交互时,计算机出现错误时的输出,而这个输出默认是和文件描述符1指向一个位置;

0x01 实例体验

既然它们是默认的,我就可以更改它们。下面的命令就是把标准输出的位置改到xlinsist文件中:

exec 1> xlinsist

这回如果我输入ls -al 或者ps命令,我们的终端将不会显示任何东西。现在,我们可以新开一个终端查看xlinsist这个文件中是否有上面两个命令所显示的内容。注意:你必须新开一个终端。

同样的道理,我们也可以改变标准输入的位置。首先,我们先看看没改变的样子:

vincent@geek:~/test$ read user
xlinsist
vincent@geek:~/test$ echo $user
xlinsist
vincent@geek:~/test$

也就是我们从键盘输入把xlinsist读入到user变量。这个read需要我输入。现在,我要改变标准输入的默认位置:

#我只是把当前的标准输出重定向到test文件中
vincent@geek:~/test$ echo 'xlinsist' 1> test
vincent@geek:~/test$ cat test 
xlinsist
#我只是把当前的标准输入重定向到test文件中
vincent@geek:~/test$ read user 0< test 
vincent@geek:~/test$ echo $user
xlinsist
vincent@geek:~/test$ 

从上面的read命令中可以看作,我并没有被要求输入什么。

标准错误输出和标准输出的区别是,它在命令出错情况下的输出。这没有什么太大的不同,我们也可以把它的输出修改到任何我们想要的位置。只不过我们需要把上面标准输出的1改成2,命令如下:

exec 2> xlinsist

当然,除了0, 1,2以外,我们可以分配自己的文件描述符。看下面的例子:

vincent@geek:~/test$ exec 6>test
vincent@geek:~/test$ echo 'i love linux shell!!!' 1>&6
vincent@geek:~/test$ cat test 
i love linux shell!!!

上面的命令很有意思:我首先把文件描述符6指向test文件。因为不像描述符1,所有的输出都会自然找它,然后看它是定向到显示器还是某个文件。所以当我们想找描述符6的时候我们要用&来引用它。其实我们可以把文件描述符想像成一个文件的引用,它可以指向任何一个文件(包括显示器),指向的过程就是我们修改默认位置的过程。而用&符号来找到它指向的目标文件,从而向其写入数据。

如果你真正了解了上面的原理后,我们就可以随便玩什么输入重定向啊、输出重定向啊,那都是小case。现在让我们来个更加复杂的例子吧,来帮你们整理一下思路,脚本如下:

exec 3>&1
exec 1>test
echo "这句话被存到test文件中"
echo "还有这句"
exec 1>&3
echo "这句话输出到显示器"

我们来一步一步理解上面的命令:首先文件描述符1默认指向的是显示器,用&来找到文件描述符1指向的目标文件,也就是显示器。因此文件描述符3也指向了显示器。然后,我们修改了文件描述符1指向的文件到test文件。接着两个echo命令的输出会自然去找文件描述符1,然后它看到文件描述符1指向的是test文件,所以它会把输出写到test文件中。最后,我们用&来找到文件描述符3指向的目标文件,也就是显示器,然后我们修改了文件描述符1指向的文件到显示器。因此,最后一个echo命令会自然的找文件描述符1然后输出到显示器上。

0x02 文件描述符相关的一些shell命令

lsof -a -p $$ -d 0,1,2

#下面为这个命令的输出
COMMAND   PID    USER   FD   TYPE DEVICE SIZE/OFF NODE NAME
bash    22609 vincent    0u   CHR 136,13      0t0   16 /dev/pts/13
bash    22609 vincent    1u   CHR 136,13      0t0   16 /dev/pts/13
bash    22609 vincent    2u   CHR 136,13      0t0   16 /dev/pts/13

下图是上面各项的含义


[Linux] 文件描述符_第1张图片
image.png

现在我用下面命令修改其标准错误输出:

vincent@geek:~/test$ exec 1> test 
vincent@geek:~/test$ lsof -a -p $$ -d 0,1,2

#新开一个终端,查看test文件,输出如下
COMMAND   PID    USER   FD   TYPE DEVICE SIZE/OFF   NODE NAME
bash    23061 vincent    0u   CHR 136,20      0t0     23 /dev/pts/20
bash    23061 vincent    1w   REG    8,2        0 525789 /home/vincent/test/test
bash    23061 vincent    2u   CHR 136,20      0t0     23 /dev/pts/20
  • /dev/null文件,这是一个很特殊的文件,你写入的任何东西都会清空。大家可以向里面写入数据试试效果。
  1. 我们可以把标准错误输出重定向到/dev/null,从而丢掉不想保存的错误信息
  2. 我们可以快速移除现有文件的数据而不用先删除文件在创建。命令如下:
cat /dev/null > test
  • Linux使用/tmp目录来存放不需要一直保留的文件,大多数Linux系统会在启动时自动删除/tmp目录中所有的文件。

下面的命令可供使用:

# 文档说文件的末尾必须至少包含3个'X',这个是在当前目录创建的
mktemp test.XXX
# 这个是在/tmp目录创建的
mktemp -t test.XXXXX
# 创建目录,在当前目录
mktemp -d test.XXXXX
  • tee命令 – 从标准输入读取,写到标准输出和文件。
# 把date产生的输出写到标准输出和文件
date | tee test
# 追加数据
date | tee -a test

0x03 管道和重定向之间的区别

管道是把一个程序的输出作为另一个程序的输入。 重定向是把输出定向到文件或者标准流。
这有2个非常详细的参考资料:
1、http://askubuntu.com/questions/172982/what-is-the-difference-between-redirection-and-pipe
2、http://ryanstutorials.net/linuxtutorial/piping.php
最好是先看第二个链接

你可能感兴趣的:([Linux] 文件描述符)