Я хочу создать символические ссылки (мягкие ссылки) из Java на машине с Windows Vista / 2008. Я доволен идеей, что мне нужно вызвать в JNI, чтобы сделать это. Однако мне нужна помощь по фактическому коду C. Какой системный вызов подходит для создания ссылки? Мы будем очень признательны за указатели на хорошую документацию по этому вопросу.





Не могли бы вы просто вызвать командную строку и использовать mklink?
Справедливо. Но есть ли какая-то конкретная причина? Я имею в виду, есть ли какие-то функции, которые вам нужны, которые недоступны через mklink или подобное?
Я упомянул это как вариант, это разумный выбор, но вы оплачиваете накладные расходы на создание процесса.
У меня уже есть DLL JNI, которую мне нужно расширить. Поэтому простой вызов API гораздо более желателен, чем создание нового процесса для вызова отдельного приложения, беспокойство о пути к этому приложению, захват его кода возврата, чтобы узнать, произошло ли создание и т. д.
Поразмыслив, я изменил свой голос "против" на голос "за". Возможно, я не хочу этого делать, но это не неправильный ответ и может быть полезен другим.
Это было в моем списке, чтобы попробовать, из моих заметок:
API:
http://msdn.microsoft.com/en-us/library/aa363866(VS.85).aspx
BOOLEAN WINAPI CreateSymbolicLink(
__in LPTSTR lpSymlinkFileName,
__in LPTSTR lpTargetFileName,
__in DWORD dwFlags
);
Некоторые примеры C#:
http://community.bartdesmet.net/blogs/bart/archive/2006/10/24/Windows-Vista-2D00-Creating-symbolic-links-with-C_2300_.aspx
Пример C++, это cnp из другой статьи, которую я читал. Я не тестировал его, поэтому используйте его с осторожностью.
typedef BOOL (WINAPI* CreateSymbolicLinkProc) (LPCSTR, LPCSTR, DWORD);
void main(int argc, char *argv[])
{
HMODULE h;
CreateSymbolicLinkProc CreateSymbolicLink_func;
LPCSTR link = argv[1];
LPCSTR target = argv[2];
DWORD flags = 0;
h = LoadLibrary("kernel32");
CreateSymbolicLink_func =
(CreateSymbolicLinkProc)GetProcAddress(h,
if (CreateSymbolicLink_func == NULL)
{
fprintf(stderr, "CreateSymbolicLinkA not available\n");
} else
{
if ((*CreateSymbolicLink_func)(link, target, flags) == 0)
{
fprintf(stderr, "CreateSymbolicLink failed: %d\n",
GetLastError());
} else
{
printf("Symbolic link created.");
}
}
}
Сказав это, я бы не стал использовать этот код :-) Я был бы склонен либо разветвлять mklink, либо смотреть на родную библиотеку из jruby / jpython (извините, я не могу найти ее в atm, так как мое сетевое соединение нестабильно). Кажется, я помню, что jruby написал библиотеку, которая объединяет различные api posix в java (думает как chown, которые требуются для соответствия ruby, но не являются кросс-платформенными). Эта библиотека используется пользователями jpython, которые кажутся ей очень довольными. Я был бы удивлен, если бы эта библиотека не предлагала поддержку символьных ссылок.
Символические ссылки в Windows создаются с помощью Функция API CreateSymbolicLink, который принимает параметры, очень похожие на аргументы командной строки, принимаемые утилита командной строки Mklink.
Предполагая, что вы правильно ссылаетесь на заголовки JNI и Win32 SDK, ваш код может быть таким же простым, как:
JNIEXPORT jboolean JNICALL Java_ClassName_MethodName
(JNIEnv *env, jstring symLinkName, jstring targetName)
{
const char *nativeSymLinkName = env->GetStringUTFChars(symLinkName, 0);
const char *nativeTargetName = env->GetStringUTFChars(targetName, 0);
jboolean success = (CreateSymbolicLink(nativeSymLinkName, nativeTargetName, 0) != 0);
env->ReleaseStringUTFChars(symLinkName, nativeSymLinkName);
env->ReleaseStringUTFChars(targetName, nativeTargetName);
return success;
}
Обратите внимание, что это просто не в моей голове, и я не имел дела с JNI много лет, поэтому я, возможно, упустил из виду некоторые тонкости выполнения этой работы ...
Нет! Я хочу напрямую использовать соответствующий вызов C / C++.