从下面的消息中,我们知道文件 /proc/sys/net/ipv4/ip_forward 中有两个字符,但是为什么 ls 只显示这个文件的大小为零?
我知道这不是磁盘上的文件,而是内存中的文件,所以有什么命令可以让我看到 /proc 中文件的实际大小吗?
root@OpenWrt:/proc/sys/net/ipv4# cat ip_forward | wc -c
2
root@OpenWrt:/proc/sys/net/ipv4# ls -l ip_forward
-rw-r--r-- 1 root root 0 Sep 3 00:20 ip_forward
root@OpenWrt:/proc/sys/net/ipv4# pwd
/proc/sys/net/ipv4
那些不是磁盘上的真正文件(正如你提到的),但它们也不是内存中的文件 - /proc 中的名称对应于对操作系统中正在运行的内核的调用,并且内容是动态生成的。
如果不生成文件,系统不知道文件有多大,但是如果您两次读取“文件”,则不能保证您获得相同的数据,因为系统可能已经更改。
您可能正在寻找该程序 sysctl -a 相反。
/proc 中的东西不是真正的文件。在大多数情况下,它们甚至不是内存中的文件。当您访问这些文件时,proc 文件系统驱动程序执行系统调用以获取适合文件的数据,然后将其格式化以供输出。这通常是即时构建的动态数据。一个例子是 /proc/net/arp,它包含当前的 ARP 缓存。
获取这些东西的大小只能通过格式化整个输出来完成,所以不是在列出文件时才完成的。如果您想要尺寸,请像以前一样使用
wc -c
。
/proc/
文件系统是由内核维护的“错觉”,它不会费心给出(大部分)其伪文件的大小(因为计算“真实”大小通常涉及构建整个文本伪文件文件的内容),并期望来自/proc/
的大多数[伪]文本文件以合理大小(例如1K)的块从第一个字节到最后一个字节(即直到EOF)按顺序读取。有关详细信息,请参见proc(5) 手册页。
所以没有办法在单个系统调用(如
stat(2),因为它会给出 0 的大小,如
报告的那样) stat(1)或 ls(1) 命令)。阅读这些文本文件的典型方式可能是
/proc/self/maps
当然,一些文件(例如/proc/sys/net/ipv4/ip_forward
)被记录为可能包含NUL
字节。你需要一些
FILE* f = fopen("/proc/self/maps", "r");
// or some other textual /proc file,
// e.g. /proc/sys/net/ipv4/ip_forward
if (f)
{
do {
// you could use readline instead of fgets
char line[256];
memset (line, 0, sizeof(line));
if (NULL == fgets(line, sizeof(line), f))
break;
// do something with line, for example:
fputs(line, stdout);
} while (!feof (f));
fclose (f);
}
给他们。
它并不是真正的内存中的文件,它是用户和内核之间的接口。