本文记录我写的一个测试代码在 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();

也就是说不应该通过文件长度来判断是否没有内容,可以尝试读取试试,如果能读取到那就证明存在内容

本文代码放在 githubgitee 上,可以使用如下命令行拉取代码

先创建一个空文件夹,接着使用命令行 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 和国产系统的开发相关博客,请参阅 博客导航


本文会经常更新,请阅读原文: https://blog.lindexi.com/post/dotnet-%E8%AE%B0%E5%9C%A8-Linux-%E4%B8%8A%E6%9F%90%E4%BA%9B%E6%96%87%E4%BB%B6%E7%9A%84%E6%96%87%E4%BB%B6%E9%95%BF%E5%BA%A6%E4%B8%BA-0-%E4%BD%86%E6%98%AF%E5%AD%98%E5%9C%A8%E5%86%85%E5%AE%B9.html ,以避免陈旧错误知识的误导,同时有更好的阅读体验。

如果你想持续阅读我的最新博客,请点击 RSS 订阅,推荐使用RSS Stalker订阅博客,或者收藏我的博客导航

知识共享许可协议 本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可。欢迎转载、使用、重新发布,但务必保留文章署名林德熙(包含链接: https://blog.lindexi.com ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请 与我联系

微软最具价值专家


无盈利,不卖课,做纯粹的技术博客

以下是广告时间

推荐关注 Edi.Wang 的公众号

欢迎进入 Eleven 老师组建的 .NET 社区

以上广告全是友情推广,无盈利