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
的枚举。同样,您对 FileShare
和 FileMode
的使用也不正确。
这个 p/invoke 声明应该足够了:http://www.pinvoke.net/default.aspx/kernel32.createfile
正如 Alexei 所说,不要调用 GetLastError
,因为您可能会获取框架调用的错误代码,而不是真正的错误。使用 SetLastError = true
和 Marshal.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());
}
}
}
我正在尝试 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
的枚举。同样,您对 FileShare
和 FileMode
的使用也不正确。
这个 p/invoke 声明应该足够了:http://www.pinvoke.net/default.aspx/kernel32.createfile
正如 Alexei 所说,不要调用 GetLastError
,因为您可能会获取框架调用的错误代码,而不是真正的错误。使用 SetLastError = true
和 Marshal.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());
}
}
}