前言
在C++的代码运行过程中,若是发生异常错误,需要抛出异常在Java代码中来处理。JNI函数中有Throw
、ThrowNew
两个方式来抛出异常。在第四节(《NDK开发 从入门到放弃(四:JNI函数、C与C++调用函数的区别)》)我们简单介绍了这两个函数,这里将以实例来稍微讲解一下。
简单实例
这里我们实现如下功能:函数获取字符串的长度,若字符串为null对象,则使用Throw
抛出NullPointerException
异常,当字符串长度为5时,使用ThrowNew
抛出IllegalArgumentException
异常,否则返回实际长度。
public class JNIDynamicUtils {
/**
* 调用C++代码的方法,当字符串为null或长度为5时抛出异常,否则,返回该字符串的长度
* @return
*/
public static native int getStrLength(String str) throws NullPointerException, IllegalArgumentException;
/**
* 加载so库或jni库
*/
static {
System.loadLibrary("JNI_DYNAMIC_ANDROID_TEST");
}
}
#include <stdio.h>
#include <jni.h>
#include <stdlib.h>
// C++层 native函数
jint getLengthOfStr(JNIEnv *env, jclass clazz, jstring str) {
if (str == NULL) {
// 抛出空指针异常
jclass clz = env->FindClass("java/lang/NullPointerException");
jmethodID methodId = env->GetMethodID(clz, "<init>", "()V");
jthrowable throwable = (jthrowable) env->NewObject(clz, methodId);
env->Throw(throwable);
} else {
jint len = env->GetStringUTFLength(str);
if (len == 5) {
// 抛出IllegalArgumentException异常
env->ThrowNew(env->FindClass("java/lang/IllegalArgumentException"), "this is IllegalArgumentException error form C++, because the str length is 5");
} else {
return len;
}
}
}
/**
* JNINativeMethod由三部分组成:
* (1)Java中的函数名;
* (2)函数签名,格式为(输入参数类型)返回值类型;
* (Ljava/lang/String;)I I表示返回值为int型,(Ljava/lang/String;)表示需要一个String参数
* (3)native函数名
*/
static JNINativeMethod gMethods[] = {
{"getStrLength", "(Ljava/lang/String;)I", (void *) getLengthOfStr }
};
//System.loadLibrary过程中会自动调用JNI_OnLoad,在此进行动态注册
JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM *jvm, void *reserved) {
JNIEnv *env = NULL;
jint result = JNI_FALSE;
//获取env指针
if (jvm->GetEnv((void**) &env, JNI_VERSION_1_6) != JNI_OK) {
return result;
}
if (env == NULL) {
return result;
}
//获取类引用
jclass clazz = env->FindClass("<包名>/JNIDynamicUtils");
if (clazz == NULL) {
return result;
}
//注册方法
if (env->RegisterNatives(clazz, gMethods, sizeof(gMethods) / sizeof(gMethods[0])) < 0) {
return result;
}
//成功
result = JNI_VERSION_1_6;
return result;
}
private void dynamicJniGetStrLength(String str) {
// tvInfo为界面上的一个TextView控件
try {
tvInfo.setText(tvInfo.getText().toString() + "\n" + "--->dynamic jni: " + JNIDynamicUtils.getStrLength(str));
} catch (NullPointerException e) {
tvInfo.setText(tvInfo.getText().toString() + "\n" + "--->dynamic jni:" + e);
} catch (IllegalArgumentException e) {
tvInfo.setText(tvInfo.getText().toString() + "\n" + "--->dynamic jni:" + e);
}
}
如上代码,我们在C++函数中,根据不同的条件抛出不同的异常或返回正常的结果。当点击按钮时,触发dynamicJniGetStrLength
方法(该方法将调用native方法,且获取返回信息显示),用如下代码测试:
dynamicJniGetStrLength(null);
dynamicJniGetStrLength("hello world");
dynamicJniGetStrLength("hello");