SetFileTime 返回错误代码 5

SetFileTime returning error code 5

我正在尝试 P/Invoke SetFileTime,但我似乎无法让它工作。对我来说,它总是 returns 错误代码 5(访问被拒绝),我不确定为什么。这是我正在使用的代码:

void Main()
{
    var pointer = CreateFile(@"C:\Users\User\Desktop\New folder\New Text Document.txt", FileAccess.ReadWrite, FileShare.None, IntPtr.Zero, FileMode.Open, FileAttributes.Normal, IntPtr.Zero);
    Console.WriteLine(pointer);
    var now = DateTime.Now.ToFileTime();
    long lpCreationTime = now;
    long lpLastAccessTime = now;
    long lpLastWriteTime = now;
    if (!SetFileTime(pointer, ref lpCreationTime, ref lpLastAccessTime, ref lpLastWriteTime))
        Console.WriteLine(GetLastError());
    CloseHandle(pointer);
}

[DllImport("kernel32.dll")]
static extern UInt32 GetLastError();

[DllImport("kernel32.dll", SetLastError = true)]
static extern Boolean SetFileTime(IntPtr hFile, ref long lpCreationTime, ref long lpLastAccessTime, ref long lpLastWriteTime);

[DllImport("kernel32.dll", SetLastError = true)]
static extern Boolean CloseHandle(IntPtr hObject);

[DllImport("kernel32.dll", CharSet = CharSet.Unicode, SetLastError = true)]
static extern IntPtr CreateFile(String fileName, [MarshalAs(UnmanagedType.U4)] FileAccess fileAccess, [MarshalAs(UnmanagedType.U4)] FileShare fileShare, IntPtr securityAttributes, [MarshalAs(UnmanagedType.U4)] FileMode creationDisposition, [MarshalAs(UnmanagedType.U4)] FileAttributes flags, IntPtr template);

我的指针有效 (2376),但我看到 SetFileTime 失败并返回错误代码 5(访问被拒绝)。现在,我已确保我是 运行 管理员,并且我的帐户有权访问该路径,但仍然没有雪茄。有人知道为什么会这样吗?

更新

Marshal.GetLastWin32Error() 也returns 5 后调用到SetFileTime。此外,我需要使此调用起作用,以便我可以 SetFileTime 在 Windows 中的长路径上,CreateFile 支持,但当前的 .NET 文件库不支持长路径在 Windows.

来自 SetFileTime 的文档:

A handle to the file or directory. The handle must have been created using the CreateFile function with the FILE_WRITE_ATTRIBUTES access right.

您的代码无法做到这一点。 .net FileAccess 枚举与 Win32 访问标志不兼容。您需要定义一个专门用于 CreateFile 的枚举。同样,您对 FileShareFileMode 的使用也不正确。

这个 p/invoke 声明应该足够了:http://www.pinvoke.net/default.aspx/kernel32.createfile


正如 Alexei 所说,不要调用 GetLastError,因为您可能会获取框架调用的错误代码,而不是真正的错误。使用 SetLastError = trueMarshal.GetLastWin32Error()

如果 CreateFile,您也无法检查 return 值中的错误。

感谢所有帮助,我能够使用以下内容实现我的目标(FileAccess.ReadWrite 转换为 0x3 而 FILE_WRITE_ATTRIBUTES 为 0x100):

[DllImport("kernel32.dll", SetLastError = true)]
static extern Boolean SetFileTime(SafeFileHandle hFile, ref long lpCreationTime, ref long lpLastAccessTime, ref long lpLastWriteTime);

[DllImport("kernel32.dll", CharSet = CharSet.Unicode, SetLastError = true)]
static extern SafeFileHandle CreateFile(String fileName, uint fileAccess, [MarshalAs(UnmanagedType.U4)] FileShare fileShare, IntPtr securityAttributes, [MarshalAs(UnmanagedType.U4)] FileMode creationDisposition, [MarshalAs(UnmanagedType.U4)] FileAttributes flags, IntPtr template);

static void Main(string[] args)
{
    var handle = CreateFile(@"C:\Users\User\Desktop\FileTimeTest.txt", (uint)(0x3 | 0x100), FileShare.None, IntPtr.Zero, FileMode.Create, FileAttributes.Normal, IntPtr.Zero);
    if (!handle.IsInvalid)
    {
        using (var stream = new FileStream(handle, FileAccess.ReadWrite))
        using (var writer = new StreamWriter(stream))
        {
            writer.WriteLine("Hello, world.");
            var now = DateTime.MaxValue.ToFileTime();
            if (!SetFileTime(handle, ref now, ref now, ref now))
                Console.WriteLine(Marshal.GetLastWin32Error());
        }
    }
}