android c java库_Android NDK — Java 与 C/C++ 的互相调用

本文使用的 native 代码以 c语言为例,c++代码类似。

这里主要介绍 C 调 Java 的两种方法,Java 调 C 比较简单。

Java 调 C

加载 so 库,"native-lib"为库名。

static {

System.loadLibrary("native-lib");

}

Java native方法

public native void test();

写一个带 native 关键字的方法,写完之后 Android Studio 会提示红色错误,直接 alt + enter 解决报错,然后会在配置好 jni 环境下会在 .c 的文件下直接生成类似下面的代码。

Java_com_example_wutao_nativedemo_NativeUtils_test(JNIEnv *env, jobject instance) {

}

方法格式:以下划线连接 "Java" + "包名" + "native方法所在的类名" + "对应的native方法名"

然后就可以这个方法体内写相应的代码逻辑了,至此 Java 调 C 就介绍完了。当然上述步骤也可以反过来完成,只不过c的方法没办法自动生成,容易写错,所以推荐按上述步骤来写,是不是很简单呢。

C调Java

方法一

可以在 Java native 方法的参数列表中传入一个 Java 对象,然后 C 里面

通过对象调方法回调 Java ,和 Java 里面常见的 setCallback 是一个道理。

具体步骤如下:

调用 setCallback 时传入实现了该接口的实例。

//自定义的一个接口

public interface OnNativeCallback {

void callbackForNative(int i);

}

...

//这里可以不是static,为了方便就这样写了

public native static void setCallback1(OnNativeCallback callback);

在native方法中回调 Java

JNIEXPORT void JNICALL

Java_com_example_wutao_nativedemo_NativeUtils_setCallback1(JNIEnv *env, jclass type, jobject callback) {

//通过传进来的对象找到该类

jclass javaClass = (*env)->GetObjectClass(env, callback);

if (javaClass == 0) {

return;

}

//获取要回调的方法ID,回调java方法

jmethodID javaCallbackId = (*env)->GetMethodID(env, javaClass, "callbackForNative", "(I)V");

(*env)->CallVoidMethod(env, callback, javaCallbackId, 123);

}

上面的调用步骤就相当于 Java 中对象调方法,当然也可以改成直接调用静态方法,那样可以不传对象,步骤也差不多,但是这种方法更符合 Java 的思维方式;这里有些地方不明白的可以看我下一片文章。

方法二

方法二是方法一的延伸,方法一只适合在当前线程中回调,就会有局限性,所以研究了一种在多线程中回调的方法,基本步骤和方法一中差别不大,只是涉及了 native中的线程的使用,对 pthread 线程不熟悉的可以查看我这篇文章Android NDK — Native 线程 pthread。

同方法一步骤一

在子线程中回调

JavaVM* java_vm = NULL;

JNIEXPORT void JNICALL

Java_com_example_wutao_nativedemo_NativeUtils_setCallback2(JNIEnv *env, jclass type, jobject callback) {

pthread_t pthread;

pthread_attr_t pthreadAttr; // 线程属性

jobject g_callback = (*env)->NewGlobalRef(env, callback);// 生成全局引用

pthread_attr_init(&pthreadAttr); //初始化线程属性

pthread_attr_setdetachstate(&pthreadAttr, PTHREAD_CREATE_DETACHED);

pthread_create(&pthread,&pthreadAttr,callbackRunnable,g_callback);// 创建线程

}

void callbackRunnable(void *callback) {

JNIEnv *env = NULL;

int ret = 0;

for (int i = 0; i < 10000; ++i) {

ret++;

}

(*java_vm)->AttachCurrentThread(java_vm,&env,NULL);

jclass javaClass = (*env)->GetObjectClass(env, callback);

if (javaClass == 0) {

return;

}

//获取要回调的方法ID,回调java方法

jmethodID javaCallbackId = (*env)->GetMethodID(env, javaClass, "callbackForNative", "(I)V");

(*env)->CallVoidMethod(env, callback, javaCallbackId, ret);

// 删除全局引用

(*env)->DeleteGlobalRef(env,callback);

(*java_vm)->DetachCurrentThread(java_vm);

//释放线程资源

pthread_exit(NULL);

}

//当.so库加载的时候会调用,用于获取JavaVM

JNIEXPORT jint JNI_OnLoad(JavaVM* vm, void* reserved) {

JNIEnv *env;

if ((*vm)->GetEnv(vm, (void**) &env, JNI_VERSION_1_4) != JNI_OK)

{

return -1;

}

java_vm = vm;

return JNI_VERSION_1_4;

}

以上就是c 回调 java 的两种方法,基本上适用于大部分情况了,如果有补充的,欢迎留言。

示例代码

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值