我需要枚举用于Windows的Java应用程序中的文件的硬链接。我考虑过将Runtime与fsutil hardlink list
一起使用,但需要管理员权限。那导致我去了WINAPI(呃)。 Kernel32.dll函数FindFirstFileNameW和FindNextFileNameW应该这样做,但是我在通过JNA API调用它时遇到了麻烦。 FindFirstFileNameW具有以下规范:
HANDLE WINAPI FindFirstFileNameW(
_In_ LPCWSTR lpFileName,
_In_ DWORD dwFlags,
_Inout_ LPDWORD StringLength,
_Inout_ PWCHAR LinkName
);
我的JNA映射:
public interface Kernel32 extends StdCallLibrary {
Kernel32 INSTANCE = (Kernel32)Native.loadLibrary("kernel32",Kernel32.class, W32APIOptions.UNICODE_OPTIONS);
HANDLE FindFirstFileNameW(WString wString, int dwFlags, IntByReference StringLength, ______ LinkName);
boolean FindNextFileNameW(HANDLE hFindStream, IntByReference StringLength, ______ LinkName);
boolean FindClose(HANDLE hFindFile);
int GetLastError();
}
呼叫功能的相关部分:
public String[] getHardLInks(Path path)
{
Kernel32 lib = Kernel32.INSTANCE;
IntByReference stringLength = new IntByReference();
______ linkName = ______;
HANDLE hFile = lib.FindFirstFileNameW(new WString(path.toString()), 0, stringLength, linkName);
String hardlink = //Convert linkName to String
//Add to array
// Loop through FindNextFileName
//Close handle
}
空白是我尝试映射LinkName
,“指向缓冲区的指针,用于存储lpFileName
”的第一个链接名称,很多事情都没有成功。这包括,字符串,char [],byte [],Buffer,Pointer和Memory,但没有任何东西从调用中返回。我相信其余的从我在测试中从StringLength得到的返回值是正确的。但是,GetLastError
总是返回ERROR_MORE_DATA
(234),根据文档意味着缓冲区太小,即使我已经传入了千字节的内存。
我应该使用什么数据类型以及如何将其恢复为字符串?
答案 0 :(得分:2)
PWCHAR
=> char[]
,虽然NIO缓冲区或内存也可以工作(请记住缓冲区和内存计数以字节为单位,而API期望以字符计数)。
您还需要将IntByReference
初始化为数组的大小,否则您告诉API您的缓冲区大小为零,并且它将顺从地填充您的缓冲区中的字符正好为零。
与大多数此类MS API函数一样,您可以将null
传递给缓冲区,然后将所需的缓冲区大小写入长度引用。
修改强>
另一件事 - 由于您(正确)使用W32API_UNICODE_OPTIONS
来初始化库映射,String
现在将自动映射到宽字符串(因此您不必使用{ {1}}),WString
函数后缀会自动处理,因此您可以从方法名称中删除-A/W
后缀。