在Android中从C ++调用Java方法


91

我试图从C ++获得一个简单的Java方法调用,而Java调用本机方法。这是Java代码:

public class MainActivity extends Activity {
    private static String LIB_NAME = "name";

    static {
        System.loadLibrary(LIB_NAME);
    }

    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        TextView tv = (TextView) findViewById(R.id.textview);
        tv.setText(this.getJniString());
    }

    public void messageMe(String text) {
        System.out.println(text);
    }

    public native String getJniString();
}

我试图messageMegetJniString*从Java到本机的方法调用过程中从本机代码调用方法。

native.cpp:

#include <string.h>
#include <stdio.h>
#include <jni.h>

jstring Java_the_package_MainActivity_getJniString( JNIEnv* env, jobject obj, jint depth ){

//    JavaVM *vm;
//    JNIEnv *env;
//    JavaVMInitArgs vm_args;
//    vm_args.version = JNI_VERSION_1_2;
//    vm_args.nOptions = 0;
//    vm_args.ignoreUnrecognized = 1;
//
//    // Construct a VM
//    jint res = JNI_CreateJavaVM(&vm, (void **)&env, &vm_args);

    // Construct a String
    jstring jstr = env->NewStringUTF("This string comes from JNI");
    // First get the class that contains the method you need to call
    jclass clazz = env->FindClass("the/package/MainActivity");
    // Get the method that you want to call
    jmethodID messageMe = env->GetMethodID(clazz, "messageMe", "(Ljava/lang/String;)V");
    // Call the method on the object
    jobject result = env->CallObjectMethod(jstr, messageMe);
    // Get a C-style string
    const char* str = env->GetStringUTFChars((jstring) result, NULL);
    printf("%s\n", str);
        // Clean up
    env->ReleaseStringUTFChars(jstr, str);

//    // Shutdown the VM.
//    vm->DestroyJavaVM();

    return env->NewStringUTF("Hello from JNI!");
}

干净的编译应用停止并显示下一条消息后:

ERROR/AndroidRuntime(742): FATAL EXCEPTION: main
        java.lang.NoSuchMethodError: messageMe
        at *.android.t3d.MainActivity.getJniString(Native Method)
        at *.android.t3d.MainActivity.onCreate(MainActivity.java:22)

显然,这意味着方法名称错误,但对我来说似乎不错。


21
请将您的解决方案作为普通答案发布,以使您的问题和解决方案更易于阅读,从而对社区更有用。您还可以与其他已经回答的人合作以完成他们的回答。
misiu_mp 2012年

@Denys:我按照您的编码进行操作,但出现此错误:java.lang.UnsatisfiedLinkError:getJniString。您能帮我解决此错误吗?
休塔2014年

@AlexTran,它已经很久了,但是从错误中判断,您可能拼写错误,或者没有getJniString在Java或C语言中链接该方法。确保通过系统导入将C代码正确链接到Java(可能现在不记得所有这些东西了:P)
Denys S.

1
这怎么从c调用java方法?这显然是Java的onCreate方法调用您的本机
约翰

使用环境(env)变量执行时,我正在获取“->”的基本操作数具有非指针类型“ JNIEnv”。如果想在没有env *变量的情况下执行操作,例如从JNI到Java层的回调,该怎么办!有任何建议!
2015年

Answers:


45

如果是对象方法,则需要将该对象传递给CallObjectMethod

jobject result = env->CallObjectMethod(obj, messageMe, jstr);

您正在做什么相当于jstr.messageMe()

由于您的方法无效,因此您应该调用:

env->CallVoidMethod(obj, messageMe, jstr);

如果要返回结果,则需要更改JNI签名(()V意味着void返回类型的方法)以及Java代码中的返回类型。


请帮我指导我,因为我的PS :)
Denys S.

我得到与您建议相同的结果。
Denys S.

1
实际上有一个CallVoidMethod,CallObjectMethod等,每个都有不同的返回类型。由于您的messageMe方法是(Ljava / lang / String;)V,因此您需要使用CallVoidMethod。
马修·威利斯,

2
请注意,你可能得到错误指示您的Java本地方法(在Java代码)可能不是void返回类型的,因此没有被发现的GetMethodID
马修·威利斯

10

Denys S.在问题中发布的解决方案:

我完全搞砸了C到C ++的转换(基本上是env可变的东西),但是我将其与C ++的以下代码一起使用:

#include <string.h>
#include <stdio.h>
#include <jni.h>

jstring Java_the_package_MainActivity_getJniString( JNIEnv* env, jobject obj){

    jstring jstr = (*env)->NewStringUTF(env, "This comes from jni.");
    jclass clazz = (*env)->FindClass(env, "com/inceptix/android/t3d/MainActivity");
    jmethodID messageMe = (*env)->GetMethodID(env, clazz, "messageMe", "(Ljava/lang/String;)Ljava/lang/String;");
    jobject result = (*env)->CallObjectMethod(env, obj, messageMe, jstr);

    const char* str = (*env)->GetStringUTFChars(env,(jstring) result, NULL); // should be released but what a heck, it's a tutorial :)
    printf("%s\n", str);

    return (*env)->NewStringUTF(env, str);
}

Java方法的下一个代码:

    public class MainActivity extends Activity {
    private static String LIB_NAME = "thelib";

    static {
        System.loadLibrary(LIB_NAME);
    }

    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        TextView tv = (TextView) findViewById(R.id.textview);
        tv.setText(this.getJniString());
    }

    // please, let me live even though I used this dark programming technique
    public String messageMe(String text) {
        System.out.println(text);
        return text;
    }

    public native String getJniString();
}

千万native方法必须是静态的?
IgorGanapolsky '16
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.