ndk c调用java_android – 如何从NDK C线程调用Java API?

我想从NDK C线程调用

Java API,但env-> FindClass()返回0.但是当我在主线程中调用Java API时,它工作得很好.我已经在线程中调用了AttachCurrentThread(),任何人都可以帮我吗?

这是源代码:

JAVA代码:

public class simple_test extends Activity {

...

// This functin will be called in C++

public void PrintNdkLog(String slog) {

Log.e(logTagNDK,slog);

return;

}

}

C代码:

static JavaVM* g_JavaVM = NULL;

jobject getInstance(JNIEnv *env,jclass obj_class)

{

jmethodID c_id = env->GetMethodID(obj_class,"","()V");

jobject obj = env->NewObject(obj_class,c_id);

return obj;

}

// JNI OnLoad

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

{

g_JavaVM = jvm;

return JNI_VERSION_1_6;

}

// Call JAVA API "PrintNdkLog" in this function

void PrintNdkLog(char *lpLog)

{

if (g_JavaVM == NULL)

return;

JNIEnv *env = NULL;

g_JavaVM->GetEnv((void**)&env,JNI_VERSION_1_6);

if (env == NULL)

return;

jclass cls = env->FindClass("com/myndk/simple_test");

if (cls != 0) // **cls will be 0 when PrintNdkLog() is called in thread**

{

LOGE("FindClass error %p",cls);

}

else

{

jmethodID mid;

jobject obj;

obj = getInstance(env,cls);

mid = env->GetMethodID(cls,"PrintNdkLog","(Ljava/lang/String;)V");

if (mid != 0)

{

jstring jstrMSG = env->NewStringUTF(lpLog);

env->CallVoidMethod(obj,mid,jstrMSG);

}

}

}

// Call JAVA API in thread

static void* thread_test(void* ptr)

{

JNIEnv *envLocal;

int status = g_JavaVM->GetEnv((void **) &envLocal,JNI_VERSION_1_6);

if (status == JNI_EDETACHED)

{

status = g_JavaVM->AttachCurrentThread(&envLocal,NULL);

if (status != JNI_OK)

LOGE("AttachCurrentThread Failed %d",status);

}

PrintNdkLog("bbb"); // This JAVA callback Failed,and printed "FindClass error"

}

// Create thread

int NdkThread(AFX_THREADPROC pfnThreadProc,LPVOID pParam,int nPriority)

{

PrintNdkLog("aaa"); // This JAVA callback runs well

pthread_t pid;

pthread_create(&pid,NULL,thread_test,pParam);

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Android NDK中,我们可以使用JNI(Java Native Interface)来实现C/C++代码和Java代码的相互调用。 下面是一个简单的例子,展示了如何在NDK调用Java层的方法。 首先,在Java层中创建一个类,并在其中声明一个需要被C/C++回调的方法: ```java public class MyCallback { public void processData(byte[] data) { // 处理数据的逻辑 } } ``` 然后,在C/C++代码中,我们需要使用JNI来获取Java层的MyCallback对象,并调用其processData方法。具体步骤如下: 1. 首先,需要在C/C++代码中引入JNI头文件: ```c++ #include <jni.h> ``` 2. 获取Java层的MyCallback对象: ```c++ JNIEnv* env; JavaVM* jvm; // 获取当前线程的JNIEnv指针 jvm->AttachCurrentThread(&env, NULL); // 获取MyCallback类 jclass myCallbackClass = env->FindClass("com/example/MyCallback"); // 获取MyCallback对象 jmethodID constructor = env->GetMethodID(myCallbackClass, "<init>", "()V"); jobject myCallbackObj = env->NewObject(myCallbackClass, constructor); ``` 3. 调用MyCallback对象的processData方法: ```c++ // 获取processData方法的ID jmethodID processDataMethod = env->GetMethodID(myCallbackClass, "processData", "([B)V"); // 构造byte[]对象 jbyteArray data = env->NewByteArray(size); env->SetByteArrayRegion(data, 0, size, (jbyte*)buf); // 调用processData方法 env->CallVoidMethod(myCallbackObj, processDataMethod, data); ``` 最后,记得在C/C++代码中释放JNI相关资源: ```c++ jvm->DetachCurrentThread(); env->DeleteLocalRef(myCallbackClass); env->DeleteLocalRef(myCallbackObj); env->DeleteLocalRef(data); ``` 以上就是在NDK中实现线程回调Java层方法的基本步骤。需要注意的是,在调用Java层方法时,需要使用JNIEnv指针。此外,如果在多线程环境下操作JNI,需要使用jvm->AttachCurrentThread()方法获取当前线程的JNIEnv指针。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值