我需要枚举用于
Windows的
Java应用程序中的文件的硬链接.我考虑过使用带有fsutil硬链接列表的Runtime,但需要管理权限.那导致我去了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),根据文档意味着缓冲区太小,即使我已经传入千字节的内存.
我应该使用什么数据类型以及如何将其恢复为字符串?
最佳答案 PWCHAR => char [],虽然NIO缓冲区或内存也可以工作(只记住缓冲区和内存计数以字节为单位,而API期望以字符计数).
您还需要将IntByReference初始化为数组的大小,否则您告诉API您的缓冲区大小为零,并且它将顺从地填充缓冲区,其中包含完全为零的字符.
像这种大多数MS API函数一样,你可以为缓冲区传递null,然后它会将所需的缓冲区大小写入你的长度引用.
编辑
另一件事 – 因为您(正确地)使用W32API_UNICODE_OPTIONS来初始化库映射,现在String将自动映射到宽字符串(因此您不必使用WString),并且-A / W函数后缀为自动处理,因此您可以从方法名称中删除-W后缀.