相关但不重复:
在 Ubuntu 中alias ll='ls -alF'
定义的Ubuntu 18.04 和 20.04 上,我看到“软盘驱动器”设备~/.bashrc
的以下输出:/dev/fd
对于ls
:
$ ls /dev/fd 0 1 2 3
对于ll
:
$ ll /dev/fd lrwxrwxrwx 1 root root 13 Nov 5 11:46 /dev/fd -> /proc/self/fd/
我可以看到/dev/fd
是一个符号链接/proc/self/fd/
,所以我ll
看到了:
$ ll /proc/self/fd/ total 0 dr-x------ 2 username username 0 Nov 8 19:01 ./ dr-xr-xr-x 9 username username 0 Nov 8 19:01 ../ lrwx------ 1 username username 64 Nov 8 19:01 0 -> /dev/pts/6 lrwx------ 1 username username 64 Nov 8 19:01 1 -> /dev/pts/6 lrwx------ 1 username username 64 Nov 8 19:01 2 -> /dev/pts/6 lrwx------ 1 username username 64 Nov 8 19:01 3 -> 'socket:[8239772]' lr-x------ 1 username username 64 Nov 8 19:01 4 -> /proc/29512/fd/
- 所有这些东西是什么意思?
- 软盘在这里是如何工作的?
- 什么是
/dev/pts/6
, 为什么0
,1
和2
都指向它? - 是什么
'socket:[8239772]'
? - 是什么
/proc/29512/fd/
? - 的历史用途和描述是
/dev/fd
什么?
有关的:
- 我之前的问题启发了我提出这个问题:'-sh:语法错误:在使用 `bash` 尝试在嵌入式 Linux 设备上进行进程替换时出现意外的“(”)
/proc
是一个文件系统,内核通过它向进程报告各种信息。它主要用于有关进程的信息,因此名称为“proc[esses]”。对于每个正在运行的进程,都有一个子目录/proc/<PID>
,其中<PID>
是进程 ID。/proc/self
是一个“神奇”的符号链接,它始终指向正在访问的进程/proc
。/proc/self/fd
报告进程打开的文件。每个条目都是一个“魔术”符号链接,其名称是文件描述符,其目标是打开的文件。从某种意义上说,链接实际上指向文件本身是很神奇的,即使通过调用获得的文件名readlink
不是有效的文件名,例如,对于没有名称的文件(例如匿名管道和套接字),以及已删除的文件。不涉及软盘。缩写“fd”代表文件描述符。
文件描述符
0
和是所有程序都希望找到1
的2
三个标准流:标准输入(stdin)、标准输出(stdout)和标准错误(stderr)。这些流由它们的编号定义:stdin 定义为文件描述符 0,其常规作用是接收用户输入或输入数据,对于 stdout(1,用户输出或输出数据)和 stderr(2,错误消息)也是如此./dev/pts/6
是一个终端。它是进程用来从该特定终端读取输入并将输出写入该特定终端的文件。当您在终端中“正常”运行程序时,stdin、stdout 和 stderr 都连接到终端。这是一个插座。文件描述符 3 没有标准角色,因此它特定于您正在使用的某些软件,可能是您的终端模拟器。如果你好奇的话,你可以看看插座的另一端是什么。
运行
ls /proc/self/fd/
时,ls
程序会打开/proc/self/fd
以读取其内容。由于文件描述符 0 到 3 在ls
启动时已经打开,并open
使用第一个可用的文件描述符,/proc/self/fd
最终在文件描述符 4 上打开。它与 PID 一起出现是因为内核不跟踪/proc/self
内部访问:它会记住它们作为特定进程的/proc
目录,然后使用正确的 PID 将其打印出来。(使用PID 命名空间,该 PID 以及路径可能会有所不同,具体取决于正在查找的进程。)当程序需要文件名并且您想要引用已经打开的文件时,它很有用。它是
/dev/stdin
,/dev/stdout
和/dev/stderr
, 分别等价于/dev/fd/0
,/dev/fd/1
和的概括/dev/fd/2
(并且在更一般的/dev/fd
出现之前就存在了)。它适用于许多 unix 变体。