本文是《The Java Native Interface Programmer’s Guide and Specification》读书笔记
在这里只讨论调用JNI方法可能会出现的异常,
在本地方法里捕捉并抛出异常
下面的例子说明怎样声明一个可能抛出异常的本地方法:
class CatchThrow {
//声明一个本地方法,并可能会抛出不合法参数异常
private native void doit()
throws IllegalArgumentException;
private void callback() throws NullPointerException {
throw new NullPointerException("CatchThrow.callback");
}
本地方法的实现为:
JNIEXPORT void JNICALL
Java_CatchThrow_doit(JNIEnv *env, jobject obj)
{
jthrowable exc;
jclass cls = (*env)->GetObjectClass(env, obj);
jmethodID mid =
(*env)->GetMethodID(env, cls, "callback", "()V");
if (mid == NULL) {
return;
}
(*env)->CallVoidMethod(env, obj, mid);
//判断是否发生了异常
exc = (*env)->ExceptionOccurred(env);
if (exc) {
/* 我们对发生的异常只是简单的打印出一条消息指明发生了什么异常,在清除这个异常的同时抛出一个新的异常 */
jclass newExcCls;
//对发生的异常进行描述
(*env)->ExceptionDescribe(env);
//清除掉发生的异常
(*env)->ExceptionClear(env);
newExcCls = (*env)->FindClass(env,
"java/lang/IllegalArgumentException");
if (newExcCls == NULL) {
/* Unable to find the exception class, give up. */
return;
}
//抛出一个新的异常
(*env)->ThrowNew(env, newExcCls, "thrown from C code");
}
}
当我们执行下面的main方法后:
public static void main(String args[]) {
CatchThrow c = new CatchThrow();
try {
c.doit();
} catch (Exception e) {
System.out.println("In Java:
" + e);
}
}
static {
System.loadLibrary("CatchThrow");
}
}
就会得到下面的输出:
java.lang.NullPointerException:
at CatchThrow.callback(CatchThrow.java)
at CatchThrow.doit(Native Method)
at CatchThrow.main(CatchThrow.java)
In Java:
java.lang.IllegalArgumentException: thrown from C code
需要注意的是通过JNI方法ThrowNew
抛出异常,并不会立刻打断本地方法的执行,因此需要编程人员明确地指定在发生异常后,控制权应该怎样转换。这是与Java里的异常处理所不同的地方,在Java里,一旦发生异常,JVM就自动将控制权交给最近的与所发生所匹配的try-catch块进行异常的处理。
我们可以通过两种方法来检查在JNI方法里是否发生了异常:
- 使用特殊的返回值来表示发生了异常(如NULL);
- 当方法1没有作用时(所返回值可能对于所调用方法是合法的),需要在本地代码里通过抛出异常来进行异常的的处理,在JNI中可以用
ExceptionOccurred
和ExceptionCheck
方法来检查是否发生了异常。
本地方法里对异常进行处理
在本地方法里对异常处理,通常有两种方法:
- 在发生异常时,直接返回,让方法的调用者进行处理;
- 用JNI方法
ExceptionClear
将异常清除掉,并执行自己的异常处理代码;
在调用大部分JNI方法前,对异常进行检查,处理,清除都是非常重要的。