本文记录我写的一个测试代码在 Linux 上踩坑的经验。在 Linux 上可能存在一些文件的文件长度是 0 但文件内里依然可以读取到内容。之前我不知道有这样的设计,导致了我大量逻辑判断文件长度为 0 就不实行,从而让运行结果不符合预期
逻辑非常简单,本文将使用读取 edid 文件作为例子,以下是我的 edid 文件所在的路径- /sys/class/drm/card0-DP-2/edid
复制代码 使用 ls -lh 下令获取 /sys/class/drm/card0-DP-2 文件夹内里的所有文件,可以看到大概如下的输出内容- lrwxrwxrwx 1 root root 0 4月 22 09:58 device -> ../../card0
- -r--r--r-- 1 root root 4.0K 4月 22 09:58 dpms
- -r--r--r-- 1 root root 0 4月 22 09:58 edid
- -r--r--r-- 1 root root 4.0K 4月 22 09:58 enabled
- -r--r--r-- 1 root root 4.0K 4月 22 09:58 modes
- drwxr-xr-x 2 root root 0 4月 22 09:58 power
- -rw-r--r-- 1 root root 4.0K 4月 22 09:58 status
- lrwxrwxrwx 1 root root 0 4月 22 09:58 subsystem -> ../../../../../../class/drm
- -rw-r--r-- 1 root root 4.0K 4月 22 09:58 uevent
复制代码 从上面输出可以看到在 linux 层获取的 edid 文件的长度也是 0 字节
但是假如此时用 cat 等工具查看,是可以获取到 edid 文件内容的
相应的,在 dotnet 这边,使用以下代码实验获取的 FileStream 的 Length 属性也是 0 长度- var file = "/sys/class/drm/card0-DP-2/edid";
- var fileStream = File.OpenRead(file);
- Console.WriteLine($"File.OpenRead {fileStream.Length}");
- fileStream.Dispose();
- fileStream = new FileStream(file, FileMode.Open, FileAccess.Read);
- Console.WriteLine($"new FileStream Length = {fileStream.Length}");
- fileStream.Dispose();
- using var safeFileHandle = File.OpenHandle(file);
- fileStream = new FileStream(safeFileHandle, FileAccess.Read);
- Console.WriteLine($"File.OpenHandle Length = {fileStream.Length}");
复制代码 有趣的是,假如使用 File.ReadAllBytes 是可以读取到内容的- if (File.ReadAllBytes(file).Length > 0)
- {
- Console.WriteLine($"读取成功");
- }
复制代码 其实在 Linux 下,即使文件长度是 0 长度,也在某些情况可以读取到内容。如下面代码,继续读取 FileStream 的内容,运行代码可以输出可以读取到内容- var fileStream = File.OpenRead(file);
- Console.WriteLine($"File.OpenRead {fileStream.Length}");
- // 似乎还可以强行读取试试看?
- // 那就读取试试
- var buffer = ArrayPool<byte>.Shared.Rent(256);
- try
- {
- var readLength = fileStream.Read(buffer.AsSpan());
- Console.WriteLine($"ReadLength={readLength}");
- }
- finally
- {
- ArrayPool<byte>.Shared.Return(buffer);
- }
- fileStream.Dispose();
复制代码 其他的创建 FileStream 也一样可以读取到,如下面代码- // 用 new FileStream 读取不到
- // 其实读取到没有长度不代表没有内容
- // Some file systems (e.g. procfs on Linux) return 0 for length even when there's content; also there are non-seekable files.
- fileStream = new FileStream(file, FileMode.Open, FileAccess.Read);
- Console.WriteLine($"new FileStream Length = {fileStream.Length}");
- buffer = ArrayPool<byte>.Shared.Rent(256);
- try
- {
- var readLength = fileStream.Read(buffer.AsSpan());
- Console.WriteLine($"ReadLength={readLength}");
- }
- finally
- {
- ArrayPool<byte>.Shared.Return(buffer);
- }
- fileStream.Dispose();
复制代码 也就是说不应该通过文件长度来判断是否没有内容,可以实验读取试试,假如能读取到那就证明存在内容
本文代码放在 github 和 gitee 上,可以使用如下下令行拉取代码
先创建一个空文件夹,接着使用下令行 cd 下令进入此空文件夹,在下令行内里输入以下代码,即可获取到本文的代码- git init
- git remote add origin https://gitee.com/lindexi/lindexi_gd.git
- git pull origin 0f8939a9c1ac51266ba472730bf6e4ccd22c34b5
复制代码 以上使用的是 gitee 的源,假如 gitee 不能访问,请更换为 github 的源。请在下令行继续输入以下代码,将 gitee 源换成 github 源进行拉取代码- git remote remove origin
- git remote add origin https://github.com/lindexi/lindexi_gd.git
- git pull origin 0f8939a9c1ac51266ba472730bf6e4ccd22c34b5
复制代码 获取代码之后,进入 BehairracercairJifelalihay 文件夹,即可获取到源代码
更多 Linux 和国产系统的开辟相关博客,请参阅 博客导航
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。 |