从C调用Java

时间:2011-06-20 17:00:31

标签: java c reference java-native-interface

我是JNI和Android NDK的新手,我想从我的一个c语言线程调用Java API。在调用Java方法时,我收到错误:

有人可以指导我,从c线程调用Java方法的最佳方法是什么?

以下代码给我,无效的JNI参考错误

JavaVM *m_vm;
jobject obj;

JNIEXPORT  jint JNI_OnLoad(JavaVM* vm, void* reserved)
{
    m_vm = vm;
    return JNI_VERSION_1_4;

}


int aWorkerThread() {
    JNIEnv      *env;
    jmethodID   mid;

    (*m_vm)->GetEnv(m_vm, (void**) &env, JNI_VERSION_1_4);
    (*m_vm)->AttachCurrentThread(m_vm, &env, NULL); 

    jclass cls      = (*env)->GetObjectClass(env, obj)
    jmethodID mid   = (*env)->GetMethodID(env, cls, "callbackfromNative", "(Ljava/lang/String;)V");

    if (mid == NULL) {
           return -1;  // method not found
    }

    const char *pchar   = "Hello From C layer";
    jstring jStr        = NULL;
    jStr = (*env)->NewStringUTF(env, pchar);
    (*env)->CallVoidMethod(env, obj, mid,jStr); 

    return 0;
}

将obj设置为全局变量

void Java_com_my_demo_init(JNIEnv * env, jobject this, jstring logThis)  
 {  
     int nErrorCode;
     jboolean isCopy;  
     const char * szLogThis = (*env)->GetStringUTFChars(env, logThis, &isCopy);  
     __android_log_print(ANDROID_LOG_DEBUG, DEBUG_TAG, "FromJava: [%s]", szLogThis); 

     obj = this; /*Lets assign obj here*/

     /*Some other c code here..*/

     (*env)->ReleaseStringUTFChars(env, logThis, szLogThis);  
 }

错误讯息:

06-20 13:12:03.239: WARN/dalvikvm(753): JNI WARNING: 0x44e83338 is not a valid JNI reference
06-20 13:12:03.239: WARN/dalvikvm(753):              in Ldalvik/system/NativeStart;.run ()V  (GetObjectClass)

...
06-20 13:12:03.279: ERROR/dalvikvm(753): VM aborting

1 个答案:

答案 0 :(得分:2)

您已经获得了对象类(cls)和类方法(mid),但实际上并没有该对象的实例。这就像拥有一个null对象并尝试调用其中一个方法。您需要使用JNI调用AllocObjectNewObject来设置obj,这就像在对象上执行“new”Java运算符一样。 AllocObject与NewObject的不同之处在于不调用对象的构造函数。