编写java
加载动态库可以用load也可以用loadLibrary,当使用loadLibrary时System.loadLibrary("FirstJNI"),去掉前缀lib和后缀so;另外,在执行时用java -Djava.library.path=路径/*.so JNIMain
public class JNIMain {
public native void printfFirst();
static {
System.load("详细路径/libFirstJNI.so");
}
public static void main(String[] args) {
new JNIMain(). printfFirst();
}}
生成JNIMain.h
javac JNIMain.java
javah JNIMain
#include#ifndef _Included_JNIMain
#define _Included_JNIMain
#ifdef __cplusplus
extern "C" {
#endif
JNIEXPORT void JNICALL Java_JNIMain_printfFirst
(JNIEnv *, jobject);
#ifdef __cplusplus
}
#endif
#endif
编写JNIMain.c
#include "JNIMain.h"
JNIEXPORT void JNICALL Java_JNIMain_printfFirst
(JNIEnv *, jobject) {
printf("My first JNI!");
}
编译c
/usr/java/jdk1.7.0_79是JAVA_HOME路径
gcc -I/usr/java/jdk1.7.0_79/include/linux -I/usr/java/jdk1.7.0_79/include -fPIC -c JNIMain.c
生成.so
gcc -shared JNIMain.o -o libFirstJNI.so
运行
java JNIMain或java -Djava.library.path=路径/*.so JNIMain
常见错误
- Linux找不到jni.h和找不到jni_md.h原因及解决方案
首先,不要怀疑是jdk出现了不完整的问题,这会让你走很多弯路,最后还解决不了问题。
以我的jdk1.7.0_75为例,两个头文件的位置分别为:
jni.h jdk1.7.0_75/include
jni_md.h jdk1.7.0_75/include/linux
解决方案
在gcc编译时,手动引入jni.h及jni_md.h两个头文件
使用gcc -I + 头文件档案位置
- java: symbol lookup error: libFirstJNI.so: undefined symbol: printfFirst 当有多个.c文件时,最好生成so时忘记加入
- Exception in thread "main" java.lang.UnsatisfiedLinkError: libFirstJNI.so: libFirstJNI.so: undefined symbol: __gxx_personality_v0 当时是有c++和c混编的,最后统一用g++。产生这种情况有很多,网上资料也多。
网友评论