jni新手笔记三:c++ 调用java

96
DON_1007
2019.03.21 14:16* 字数 566

通过jni可以实现c/c++调用java代码,上一篇 jni新手笔记二中,定义了一个javaNDKTest用于实现java代码调用c++代码,编译之后,通过javah -jni 命令生成了一个jni的头文件 com_don_ndk_NDKTest.h

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class com_don_ndk_NDKTest */

#ifndef _Included_com_don_ndk_NDKTest
#define _Included_com_don_ndk_NDKTest
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     com_don_ndk_NDKTest
 * Method:    getHello
 * Signature: (I)Ljava/lang/String;
 */
JNIEXPORT jstring JNICALL Java_com_don_ndk_NDKTest_getHello
  (JNIEnv *, jobject, jint);

#ifdef __cplusplus
}
#endif
#endif

  • JNIEnv 表示java环境
  • jobject表示 NDKTest的实例对象
  • jint表示java调用的时候给的参数
    通过 JNIEnvjobject 可以实现c/c++ 调用java代码

一、c++调用类成员方法

NDKTest类中新增一个类成员方法printDescription()

package com.don.ndk;

import android.util.Log;

public class NDKTest {
    private final static String TAG = "NDKTest";

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

    public native String getHello(int count);

    /**
     * 供c/c++代码调用的一个类成员方法
     */
    public void printDescription() {
        Log.i(TAG, "this is a NDKTest object" + toString());
    }
}

ndktest.cpp中新增方法callObjectMethod用于调用java方法printDescription,具体实现如下

//
// Created by Don on 2019-03-20.
//
#include "com_don_ndk_NDKTest.h"
#include "hello.h"

int callObjectMethod(JNIEnv *, jobject);

JNIEXPORT jstring JNICALL Java_com_don_ndk_NDKTest_getHello
        (JNIEnv *pEnv, jobject obj, jint param) {

    jstring result = pEnv->NewStringUTF(getHello());

    callObjectMethod(pEnv, obj);
    return result;
}


/**
 * 调用 类成员方法
 * @return
 */
int callObjectMethod(JNIEnv *pEnv, jobject obj) {
    jclass ndktestclass = pEnv->GetObjectClass(obj);
    // 获取方法printDescription 方法id,
    jmethodID methodID = pEnv->GetMethodID(ndktestclass, "printDescription", "()V");
    pEnv->CallVoidMethod(obj, methodID);
    return 0;
}

因为java方法printDescription没有返回值,所以需要通过 JNIEnvCallVoidMethod方法来调用,如果java方法有返回值,就要根据其返回的值得类型来选择使用JNIEnv的方法。

CallVoidMethod的第一个参数是java对象实例,这里是从 Java_com_don_ndk_NDKTest_getHello中得到的obj,第二个参数是printDescription的方法ID方法ID需要通过JNIEnvGetMethodID获取,第一个参数是java类,第二个参数是java 方法名,第三个参数是方法的签名,方法的签名是指定该方法的参数类型及返回值类型,java类中定义的printDescription方法没有参数,没有返回值,所以这里第三个参数是 ()V

方法签名的格式需要根据java方法的参数来写,()中填写参数类型,()后面是返回值类型,比如printDescription没有参数,()中是空的,没有返回值,()后面是V;假设我们定义一个java方法 int signMethodTest(String param1),那么signMethodTest的方法签名就应该是 (Ljava/lang/String;)I

二、c++调用类静态方法

NDKTest中新增方法notifyJava

package com.don.ndk;

import android.util.Log;

public class NDKTest {
    private final static String TAG = "NDKTest";

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

    public native String getHello(int count);

    /**
     * 供c/c++代码调用的一个类成员方法
     */
    public void printDescription() {
        Log.i(TAG, "this is a NDKTest object" + toString());
    }

    /**
     * 供c/c++代码调用的一个类静态方法
     */
    public static void notifyJava(String msg) {
        Log.i(TAG, "notifyJava msg: " + msg);
    }

}

jni中新增方法 callStaticMethod 调用 NDKTestnotifyJava方法

//
// Created by Don on 2019-03-20.
//
#include "com_don_ndk_NDKTest.h"
#include "hello.h"

int callStaticMethod(JNIEnv *, jobject);

JNIEXPORT jstring JNICALL Java_com_don_ndk_NDKTest_getHello
        (JNIEnv *pEnv, jobject obj, jint param) {

    jstring result = pEnv->NewStringUTF(getHello());

    callStaticMethod(pEnv, obj);
    return result;
}

/**
 * 调用 类静态方法
 * @param pEnv
 * @param obj
 * @return
 */
int callStaticMethod(JNIEnv *pEnv, jobject obj) {
    jclass ndktestclass = pEnv->GetObjectClass(obj);
    jmethodID methodID = pEnv->GetStaticMethodID(ndktestclass, "notifyJava",
                                                 "(Ljava/lang/String;)V");
    pEnv->CallStaticVoidMethod(ndktestclass, methodID, pEnv->NewStringUTF("new msg from jni"));
    return 0;
}

调用类静态方法与调用类成员方法类似,区别是调用的是 CallStaticVoidMethod方法,第一个参数给的是java类,第二个参数方法ID是通过GetStaticMethodID获取,notifyJava方法中我们定义了一个参数,所以第三个参数是调用notifyJava传递的参数

Android