【问题标题】:Error calling java function from native code从本机代码调用 java 函数时出错
【发布时间】:2013-01-30 02:25:20
【问题描述】:

我想从本机代码调用 java 方法。我按照以下步骤操作: Calling a java method from c++ in Android

但是我的应用程序在调用 java 函数的地方崩溃了。

我的java代码如下图:

package com.example.jnitry;

import android.os.Bundle;
import android.app.Activity;
import android.util.Log;
import android.view.Menu;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;

public class MainActivity extends Activity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        Log.d("MYAPP","Ocreate entered....");
        Button btn=(Button) findViewById(R.id.button1);
        btn.setOnClickListener(new OnClickListener() {

            @Override
            public void onClick(View arg0) {
                // TODO Auto-generated method stub
                mainfunc(8);
            }
        });
    }

    public void display(byte[] byt){
        Log.d("MYAPP", "display() is entered....");
        for(int i=0;i<byt.length;i++)
            Log.d("MYAPP", "byt["+i+"]="+byt[i]);
        Log.d("MYAPP", "display finished");
    }

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.activity_main, menu);
        return true;
    }
    public static native void mainfunc(int n);

    static
    {
        System.loadLibrary("mylib");
    }
}

我的原生代码如下图:

#include <jni.h>
#include "com_example_jnitry_MainActivity.h"
#include <android/log.h>
#include <stdio.h>

JNIEXPORT void JNICALL Java_com_example_jnitry_MainActivity_mainfunc
  (JNIEnv *env, jclass obj, jint n){

    __android_log_print(ANDROID_LOG_DEBUG,"MYAPP","mainfunction entered...",NULL);
    int i;
    unsigned char arr[10];
    jbyteArray bArray=(*env)->NewByteArray(env,n);

    jclass cls = (*env)->FindClass(env, "com/example/jnitry/MainActivity");
        jmethodID mid = (*env)->GetMethodID(env, cls, "display", "([B)V");
        if (mid == 0){
            __android_log_print(ANDROID_LOG_DEBUG,"MYAPP","mid==0",NULL);
            return;
        }
        if(bArray==NULL)
            {   __android_log_print(ANDROID_LOG_DEBUG,"MYAPP","bArray==NULL...",NULL);
                return ;
            }
    for(i=0;i<n;i++){
        arr[i]=i;
        __android_log_print(ANDROID_LOG_DEBUG,"MYAPP","Iteration number:%d",i);
    }

    (*env)->SetByteArrayRegion(env,bArray,0,n,arr);
    __android_log_print(ANDROID_LOG_DEBUG,"MYAPP","bArray successfully created...",NULL);
    (*env)->CallVoidMethod(env, obj, mid, bArray);
    __android_log_print(ANDROID_LOG_DEBUG,"MYAPP","Returned from disp() java function.....",NULL);

}

我的应用在打印日志消息后崩溃:

02-14 15:37:06.814: D/MYAPP(11584): bArray successfully created...

谁能告诉我原因。并为此提供解决方案。提前致谢。

注意:我已经尝试过CallVoidMedthodA()callByteMethod()CallObjectMethod(),但他们结果是一样的。

【问题讨论】:

    标签: android android-ndk java-native-interface


    【解决方案1】:

    AFAIK CallVoidMethod 不采用 env 参数。至少在 C++ 中。

    【讨论】:

    • 以上代码用于c。所以在每个 jni 函数调用中传递 env 作为第一个参数是强制性的。
    【解决方案2】:

    我可以给你这个。有用。但它是 C++。 (所以吃了标签——抱歉。)

    // it returns NULL in the case of an exception
    // the returned memory is calloc()'d; it's the caller's responsibility to free() it.
    char* changeEncoding(const char*source, int len, int direction)
    {
        JNIEnv* env = threadUnsafeInfo.env;
        jobject obj = threadUnsafeInfo.obj;
    
        if (!source) {
        JNU_ThrowByName(env, "java/lang/NullPointerException", 0);
        return NULL;
        }
        jbyteArray srcArray = env->NewByteArray(len);
    
        jclass cls = env->FindClass("com/xxx/Yyy");
        jmethodID mid = env->GetMethodID(cls, "convert", "([BI)[B");
    
        if (mid != NULL && srcArray != NULL) {
        env->SetByteArrayRegion(srcArray, 0, len, (jbyte*)source);
        env->ExceptionClear();
    
        jbyteArray resArray = (jbyteArray)env->CallObjectMethod(obj, mid, srcArray, direction);
        if(env->ExceptionOccurred()) {
            DLOG("exception in convert ([BI)[B");
            env->ExceptionDescribe();
            //env->ExceptionClear(); // ??
            return NULL;
        }
    
        int resultLen = env->GetArrayLength(resArray);
        char* result = (char*)calloc(2 + resultLen,1); // why 2: a bit of healthy paranoia ain't gonna hurt anyone
        if (result == 0) {
            JNU_ThrowByName(env, "java/lang/OutOfMemoryError", 0);
            return NULL;
        }
        env->GetByteArrayRegion(resArray, 0, resultLen, (jbyte *)result);
        env->DeleteLocalRef(cls);
        env->DeleteLocalRef(resArray);
        env->DeleteLocalRef(srcArray);
        return result;
        } else {
        JNU_ThrowByName(env, "java/lang/NullPointerException", 0);
        myassert(("method id = 0",0));
        }
        return NULL;
    }
    

    【讨论】:

      【解决方案3】:

      您的本机方法 mainFunc 是静态方法 - 它没有有效的 this 指针。同时,display 是一个实例方法——它需要一个。

      mainFunc 的第二个参数实际上是MainActivity 的类对象指针,根据 JNI 规则。在 Java 端将 mainFunc 声明为非静态的,这将起作用。

      【讨论】:

      • 非常感谢...按照您的建议,删除 static 关键字有效....
      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2014-07-25
      • 1970-01-01
      相关资源
      最近更新 更多