Java调用C
例子:java端调用c端的add,两个数相加
1.java端
void test() {
int addr = add(100,200);
test.setText("add ="+addr);
}
static {
System.loadLibrary("javacallc");
}
private native int add(int a,int b);
2.建jni目录(main目录下面),在java下产生头文件,拷到jni下面
javah -jni com.example.ccalljava.Jni
3.实现.C
jint Java_com_example_chenhong_ndktest_MainActivity_add(JNIEnv *env, jobject bject, jint a, jint b) {
return a + b;
}
完成,运行即可
C调用java
C语音要调用java里面的方法,需要有以下几个步骤
1.找到class
jclass (*FindClass)(JNIEnv*, const char*)
2.获取方法
jmethodID (*GetMethodID)(JNIEnv*, jclass, const char*, const char*);
3.实例化对象
jobject (*AllocObject)(JNIEnv*, jclass);
4.调用对象的方法
(*CallIntMethod)(JNIEnv*, jobject, jmethodID, ...);
看下面几个例子:
1.C调用java的add()
2.C调用java的helloString
3.C调用java带参数的helloString(str)
jint Java_com_example_ccalljava_Jni_add(JNIEnv * env, jobject obj) {
// 第二个参数为类的全路径
jclass clazz = (*env)->FindClass(env, "com/example/ccalljava/Jni");
// 第二第三个参数分别是方法名称,以及方法签名javap -s com.example.ccalljava.Jni
jmethodID methodid = (*env)->GetMethodID(env, clazz, "add", "(II)I");
jobject obj1 = (*env)->AllocObject(env, clazz);
jint result = (*env)->CallIntMethod(env, obj1, methodid, 99, 1);
return result;
}
void Java_com_example_ccalljava_Jni_callhello(JNIEnv * env, jobject obj){
jclass clazz = (*env)->FindClass(env, "com/example/ccalljava/Jni");
// 第二第三个参数分别是方法名称,以及方法签名javap -s com.example.ccalljava.Jni
jmethodID methodid = (*env)->GetMethodID(env, clazz, "helloString", "()V");
jobject obj1 = (*env)->AllocObject(env, clazz);
(*env)->CallVoidMethod(env, obj1, methodid);
}
void Java_com_example_ccalljava_Jni_callhelloString(JNIEnv * env, jobject obj, jstring jst){
jclass clazz = (*env)->FindClass(env, "com/example/ccalljava/Jni");
// 第二第三个参数分别是方法名称,以及方法签名javap -s com.example.ccalljava.Jni
jmethodID methodid = (*env)->GetMethodID(env, clazz, "helloString", "(Ljava/lang/String;)V");
jobject obj1 = (*env)->AllocObject(env, clazz);
// 生成一个jstring
jstring jst1 = (*env)->NewStringUTF(env, "hello,I.m from c");
(*env)->CallVoidMethod(env, obj1, methodid,jst1);
}
调用静态方法
对于静态方法也是一样的步骤,不同的是调用不同,且不要实例化对象,这个用java的思想很好理解。
// 获取方法
jmethodID (*GetStaticMethodID)(JNIEnv*, jclass, const char*, const char*);
// 调用方法
void (*CallStaticVoidMethod)(JNIEnv*, jclass, jmethodID, ...);
注意:
1.jni目录一定要建main下面,与java是同级目录,不然不能生成.so
2.(*env)->FindClass,类的路径com.example.ccalljava.Jni写成 “com/example/ccalljava/Jni”;
3.方法的签名通过.class字节码反编译获得的,需要在dubug目录执行:javap -s com.example.ccalljava.Jni生成()
4.3正常进行的条件是工程已经build成功,产生了字节码,不然会报错找不到该类
5.AS运行build不成功,报错如下:
是因为打开了cmd,没有关闭,导致AS编译的时候不能删除debug下面的文件,关闭cmd窗口即可。
直接使用提供的SO
对于已经编译好的SO,我们拿来使用只需要,在AS中的main()下面建立一个jniLibs目录,然后将相应的SO拷贝到jniLibs,不需要在配置build.gradle,以及c,h文件等。(对于AS如果运行的so在不支持的手机平台,那么会装不上,对于Eclipse装上后运行会崩溃)。