FFmpeg: java.lang.UnsatisfiedLinkError при вызове класса Runnable
Мне нужно взять файл изображения и аудиофайл и создать видео. Я знаю, что это можно сделать с помощью
Runtime.getRuntime().exec("ffmpeg -i image.jpeg -i audio.mp3 out.avi")
но только для рутованных устройств, поэтому я попытался создать оболочку JNI для main() из ffmpeg.c и вызвать ее из своей Activity, как здесь: http://demo860.blogspot.com/2010/07/android-ffmpeg-dynamic-module-jni.html
1.Этот код находится в ffmpeg.c:
int m_argc = 0;
char *m_pargv [30];
int dynamic_ffpmeg_main (int argc, char **argv);
jint JNICALL Java_com_ccmedia_codec_ffmpeg_mod_1run ( JNIEnv *, jclass, jstring, jstring );
jint JNICALL Java_com_ccmedia_codec_ffmpeg_mod_1run ( JNIEnv *env, jclass class, jstring pj1, jstring pj2)
{
// as in http://demo860.blogspot.com/2010/07/android-ffmpeg-dynamic-module-jni.html
}
int dynamic_ffpmeg_main(int argc, char **argv)
{
// as in http://demo860.blogspot.com/2010/07/android-ffmpeg-dynamic-module-jni.html
}
int main(int argc, char **argv)
{
dynamic_ffpmeg_main ( argc, argv );
return 0;
}
2. Этот код в моей.java:
public class FFmpegCreator implements Runnable {
static boolean m_bret = false;
static String m_szconfig = " -i /sdcard/file.mpg -vcodec mpeg4 aaa.mpg";
//public native String unimplementedStringFromJNI();
static {
try {
System.out.println("[AdDBCache] Module load try ffmpeg : "
+ System.getProperty("java.library.path"));
// System.load("/sdcard/arm_and/bin/libffmpeg.so");
System.loadLibrary("ffmpeg");
System.out.println("[AdDBCache] Module load success");
}
catch (Exception e) {
System.out.println("[AdDBCache] Module load err : "
+ System.getProperty("java.library.path"));
}
}
private static synchronized final native int Java_com_ccmedia_codec_ffmpeg_mod_1run(String name, String sztoken);
public void set_config(String sz_config) {
m_szconfig = sz_config;
}
public void run_core(String sz_file, String sz_token) {
int n_stat;
m_bret = false;
n_stat = Java_com_ccmedia_codec_ffmpeg_mod_1run(m_szconfig, sz_token);
m_bret = true;
}
public void run() {
run_core("", "");
}
}
3. И это в моей деятельности:
FFmpegCreator f = new FFmpegCreator ();
new Thread(f).start();
Но у меня есть
E/AndroidRuntime(25682): java.lang.UnsatisfiedLinkError: Java_com_ccmedia_codec_ffmpeg_mod_1run .
И я не могу понять, почему... Сборка FFmpeg была успешной... Может кто-нибудь помочь мне, пожалуйста? Я буду очень признателен, если вы поможете мне. Спасибо.
1 ответ
Проблема заключается в наименовании нативного метода Java_com_ccmedia_codec_ffmpeg_mod_1run
на стороне Java. Вы должны просто дать ему нормальное имя метода, без всех частей Java_package.... Затем, чтобы сопоставить это с функцией C, вам нужно использовать пакет и класс, которому принадлежит метод. Самый надежный способ сделать это - сначала обновить сторону Java:
public class FFmpegCreator implements Runnable {
// ...
private static synchronized final native int mod_1run(String name, String sztoken);
//...
}
а потом запусти джаву по классу:
$ javah -o FFmpegCreator.h -classpath bin/classes com.yourpackage.FFmpegCreator
(заменить bin/classes
с каталогом, где ваши файлы.class компилируются, и com.yourpackage
с пакетом, который FFmpegCreator
внутри). Если вы посмотрите на FFmpegCreator.h
он генерирует это будет включать правильную подпись для вашего собственного метода.