gpt4 book ai didi

c# - 句柄不支持与 CreateFile 的同步操作

转载 作者:行者123 更新时间:2023-11-30 15:37:26 25 4
gpt4 key购买 nike

我需要以原始模式打开读取磁盘,因此我为此目的使用了 CreateFile API 函数。

private static FileStream OpenDisk(string drive)
{
// Try to open hard disk drive in raw mode for reading
SafeFileHandle safeHandle = Native.CreateFile(
string.Format(@"\\.\{0}", drive),
FileAccess.Read,
FileShare.Read,
IntPtr.Zero,
FileMode.Open,
FileAttributes.ReadOnly | FileAttributes.Device,
IntPtr.Zero);

// Check if the drive was successfully opened
if (safeHandle.IsInvalid)
{
Marshal.ThrowExceptionForHR(Marshal.GetHRForLastWin32Error());
}

// Create file stream on the file for reading
return new FileStream(safeHandle, FileAccess.Read);
}

但是当我尝试从流中读取时出现以下错误

句柄不支持同步操作。可能需要更改 FileStream 构造函数的参数以指示句柄是异步打开的(即,它是为重叠 I/O 显式打开的)。

这是重现此问题的示例代码

using (FileStream stream = OpenDisk("X:"))
{
byte[] buffer = new byte[1000];
while (stream.Read(buffer, 0, 1000) > 0) { }
}

我真的不明白它想从我这里得到什么?当我使用更大的缓冲区(例如 4096)时,它会起作用。当我说它有效时,我的意思是它真的(总是)有效,它工作了一段时间,直到我更改缓冲区大小。我猜它在内部执行某种异步缓冲,当我指定比默认缓冲大小时更大的缓冲区时,它只是没有被使用,但如何摆脱它?

谢谢

更新

当我尝试使用 BufferedStream 读取它时,我遇到了同样的问题

using (BufferedStream stream = new BufferedStream(OpenDisk("X:"), 4096))
{
byte[] buffer = new byte[1000];
while (stream.Read(buffer, 0, 1000) > 0) { }
}

我是否理解 BufferedStream 的错误用途?它不应该读取和缓存指定大小的 block 吗?

最佳答案

简而言之,从卷中读取是非缓冲的,并且必须是整个扇区。缓冲区大小为 4096 可实现此目的。该错误消息具有误导性,因为异步 I/O 和非缓冲往往会结合在一起。一些细节:

来自CreateFile docs :

即使在 CreateFile 中未指定非缓存选项,卷句柄也可以根据特定文件系统的判断以非缓存方式打开。您应该假定所有 Microsoft 文件系统都将卷句柄打开为非缓存的。对文件的非缓存 I/O 的限制也适用于卷。

来自File Buffering docs :

  • 文件访问大小,包括 OVERLAPPED 结构中的可选文件偏移量(如果已指定),其字节数必须是卷扇区大小的整数倍。例如,如果扇区大小为 512 字节,则应用程序可以请求读取和写入 512、1,024、1,536 或 2,048 字节,但不能请求 335、981 或 7,171 字节。
  • 用于读取和写入操作的文件访问缓冲区地址应该是物理扇区对齐的,这意味着内存中的地址是卷的物理扇区大小的整数倍。根据磁盘的不同,可能不会强制执行此要求。

关于c# - 句柄不支持与 CreateFile 的同步操作,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/12608190/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com