0

我正在尝试使用该函数获取FILE_ID_BOTH_DIR_INFO 。GetFileInformationByHandle调用后我的所有值都设置为零。
我收到 win32 错误消息:
ERROR_BAD_LENGTH 程序发出命令但命令长度不正确。

IntPtr handle = CreateFile(dir, EFileAccess.GenericRead, EFileShare.Read, IntPtr.Zero, ECreationDisposition.OpenExisting, EFileAttributes.BackupSemantics | EFileAttributes.Normal, IntPtr.Zero);
FILE_ID_BOTH_DIR_INFO fileStruct = new FILE_ID_BOTH_DIR_INFO();
GetFileInformationByHandleEx(handle, FILE_INFO_BY_HANDLE_CLASS.FileIdBothDirectoryInfo, out fileStruct, (uint)Marshal.SizeOf(fileStruct));

FILE_ID_BOTH_DIR_INFO 结构

    typedef struct _FILE_ID_BOTH_DIR_INFO {
    DWORD         NextEntryOffset;
    DWORD         FileIndex;
    LARGE_INTEGER CreationTime;
    LARGE_INTEGER LastAccessTime;
    LARGE_INTEGER LastWriteTime;
    LARGE_INTEGER ChangeTime;
    LARGE_INTEGER EndOfFile;
    LARGE_INTEGER AllocationSize;
    DWORD         FileAttributes;
    DWORD         FileNameLength;
    DWORD         EaSize;
    CCHAR         ShortNameLength;
    WCHAR         ShortName[12];
    LARGE_INTEGER FileId;
    WCHAR         FileName[1];
    } FILE_ID_BOTH_DIR_INFO, *PFILE_ID_BOTH_DIR_INFO

在 C# 中

    public struct FILE_ID_BOTH_DIR_INFO
    {
        uint NextEntryOffset;
        uint FileIndex;
        LARGE_INTEGER CreationTime;
        LARGE_INTEGER LastAccessTime;
        LARGE_INTEGER LastWriteTime;
        LARGE_INTEGER ChangeTime;
        LARGE_INTEGER EndOfFile;
        LARGE_INTEGER AllocationSize;
        uint FileAttributes;
        uint FileNameLength;
        uint EaSize;
        char ShortNameLength;
        [MarshalAsAttribute(UnmanagedType.ByValTStr, SizeConst = 12)]
        string ShortName;
        LARGE_INTEGER FileId;
        [MarshalAsAttribute(UnmanagedType.ByValTStr, SizeConst = 1)]
        string FileName;
    }

    [StructLayout(LayoutKind.Explicit, Size = 8)]
    struct LARGE_INTEGER
    {
        [FieldOffset(0)]
        public Int64 QuadPart;
        [FieldOffset(0)]
        public UInt32 LowPart;
        [FieldOffset(4)]
        public Int32 HighPart;
    }
4

1 回答 1

2

ByValTStr 的字符集由 StructLayout 属性的 CharSet 参数确定。由于您没有指定,因此使用默认的 8 位 ANSI。如果您将 CharSet 指定为 CharSet.Unicode 那么应该可以解决问题。将此属性添加到您的结构中:

[StructLayout(LayoutKind.Sequential, 
    CharSet=CharSet.Unicode)]

为了调试的目的,编写一个输出结构大小的 C++ 程序。确保您的 C# 程序与该值匹配。

于 2013-03-22T07:57:25.943 回答