栏目分类:
子分类:
返回
名师互学网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
名师互学网 > IT > 软件开发 > 后端开发 > Java

【混合编程jni 】第四篇之引用和异常

Java 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

【混合编程jni 】第四篇之引用和异常

继续写JNI的知识点

上篇基本上介绍了数据的转换以及方法签名的相关知识点,不懂的可以看看之前的文章

建议循序渐进,不可冒进

今天继续介绍JNI的知识点

 除 八种 基本数据类型之外的都是引用数据类型 ;

关于引用

Java虚拟机的内存结构我们都知道,堆内存和堆外内存

大家都知道,Java代码创建的对象大多在堆内存内

Native 代码创建的对象,占用的内存在native 内存,

在混合编程的时候,对象有可能创建在Java侧,也有可能创建的native侧

但是在混合编程的时候需要做数据的传递,如果只是简单的拷贝就没什么问题

但是我们知道大对象拷贝起来性能很差,所以不能直接拷贝,还是谁创建,谁管理

但是JVM 是有gc 存在的,这个是自动回收,但是有可能在Native代码运行的时候导致对象被回收,而产生错误

因此存在几种引用对象

① 局部引用 Local Reference: 其只在作用域内有效 , 内存不可回收 ;

② 全局引用 Global References: 全局有效 , 内存不可回收 ;

③ 全局弱引用 Weak Global References: 全局有效 , 内存不足时会被 JVM 回收 ;

Global References 全局引用

先看下Api 定义

// 创建一个全局引用
jobject NewGlobalRef(JNIEnv *env, jobject obj);
//    删除一个全局引用
void DeleteGlobalRef(JNIEnv *env, jobject globalRef);

Global Reference具有全局性,可以在多个Native Method调用过程和多线程之间共享其指向的对象,

在主动调用DeleteGlobalRef之前,它是一直存在的(GC不会回收其内存)

举个例子:

class Union {
public:
    Union(JNIEnv* env, jstring s) {
        this->s = env->NewGlobalRef(s);
    }
    ~Union() {
        assert(s == NULL);
    }
    void destroy(JNIEnv* env) {
        env->DeleteGlobalRef(s);
        s = NULL;
    }
    jstring s;
};
Local Reference 本地引用

看下Api

//    创建一个新的本地引用,该引用引用与 相同的对象 ref。给定的ref可能是全局或本地引用。NULL如果ref 引用则返回null。
jobject NewLocalRef(JNIEnv *env, jobject ref);

voi DeleteLocalRef(jobject localRef);

每当在 Native代码中引用到一个Java对象时,JVM 就会在这个Table中创建一个Local Reference。

Local Reference并不是Native里面的局部变量,局部变量存放在堆栈中,而Local Reference存放在Local Reference Table中。

在离开native方法的时候,本地引用会自动释放。

注意:默认引用类型都是本地引用

异常处理

Java编程中经常遇到各种异常,好在java有异常的处理机制,try catch finaly 一套连招下来,

基本上可以解决80% 的异常情况。

JNI编程中可能也会遇到异常,native代码调用Java代码时抛出异常。

ExceptionCheck:检查是否发生了异常,若有异常返回JNI_TRUE,否则返回JNI_FALSE 
ExceptionOccurred:检查是否发生了异常,若用异常返回该异常的引用,否则返回NULL 
ExceptionDescribe:打印异常的堆栈信息 
ExceptionClear:清除异常堆栈信息 
ThrowNew:在当前线程触发一个异常,并自定义输出异常信息 
Throw:丢弃一个现有的异常对象,在当前线程触发一个新的异常 
FatalError:致命异常,用于输出一个异常信息,并终止当前VM实例(即退出程序)
工具函数

JNI编程中经常需要抛出异常,所以提供一个通用的工具函数

void ThrowByName(JNIEnv *env, const char *name, const char *msg)
{
    jclass cls = (*env)->FindClass(env, name);
    
    if (cls != NULL) {
        (*env)->ThrowNew(env, cls, msg);
    }
    
    (*env)->DeleteLocalRef(env, cls);
}

来个例子

//异常捕获 ,检查JNI调用是否有异常
if(env->ExceptionCheck()){
    env->ExceptionDescribe();
    env->ExceptionClear();//清除引发的异常,在Java层不会打印异常堆栈信息,如果不清除,后面的调用ThrowNew抛出的异常堆栈信息会
//覆盖前面的异常信息
    jclass cls_exception = env->FindClass("java/lang/Exception");
    env->ThrowNew(cls_exception,"call java static method ndk error");
    return;
}
总结:

这篇文章主要讲了Java的引用和异常

引用是为了避免Jvm GC的清理,相当于在JVM 内锁定内存

异常将C++中的检查异常留给Java回调

总之都是为了解决问题,如果不明白的时候可以思考下

如果是自己该如何操作

转载请注明:文章转载自 www.mshxw.com
本文地址:https://www.mshxw.com/it/759961.html
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 MSHXW.COM

ICP备案号:晋ICP备2021003244-6号