我正在尝试P / Invoke SetFileTime,但我似乎无法让它工作。它总是为我返回错误代码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()
后, SetFileTime
也会返回5。此外,我需要让这个调用工作,以便SetFileTime
在CreateFile
支持的Windows中的长路径上,但.NET的当前文件库不支持Windows中的长路径。 / p>
答案 0 :(得分:5)
来自SetFileTime
的文档:
文件或目录的句柄。必须使用具有FILE_WRITE_ATTRIBUTES访问权限的CreateFile函数创建句柄。
您的代码无法做到这一点。 .net FileAccess
枚举与Win32访问标志不兼容。您需要定义一个专门用于CreateFile
的枚举。同样,您对FileShare
和FileMode
的使用也不正确。
这个p / invoke声明应该足够了:http://www.pinvoke.net/default.aspx/kernel32.createfile
正如阿列克谢所说,不要调用GetLastError
,因为您可能正在为框架调用而不是真正的错误找到错误代码。使用SetLastError = true
和Marshal.GetLastWin32Error()
。
如果CreateFile
,您也无法检查返回值中的错误。
答案 1 :(得分:1)
感谢所有帮助,我能够使用以下内容实现目标(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());
}
}
}