通过jni
可以实现c/c++
调用java
代码,上一篇 jni新手笔记二中,定义了一个java
类NDKTest
用于实现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
调用的时候给的参数
通过JNIEnv
和jobject
可以实现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
没有返回值,所以需要通过 JNIEnv
的 CallVoidMethod
方法来调用,如果java
方法有返回值,就要根据其返回的值得类型来选择使用JNIEnv
的方法。
CallVoidMethod
的第一个参数是java
对象实例,这里是从 Java_com_don_ndk_NDKTest_getHello
中得到的obj
,第二个参数是printDescription
的方法ID
,方法ID
需要通过JNIEnv
的GetMethodID
获取,第一个参数是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
调用 NDKTest
的notifyJava
方法
//
// 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
传递的参数
网友评论