android jni tips,Jni Tips

属性或者方法的引用或者IDs需要好几个字符串比较,一旦你拥有它们,你执行它们(指的是方法或者对象)的速度是非常快的。

Find Class、Get methodID fieldID需要花费大量的时间来做字符串比较,推荐的做法是事先缓存这些IDs,以加快调用速度。因为Android限制每个进程只有一个JavaVM,因此在方法内定义static局部变量来缓存这些IDs可有效的提升性能。

类的引用,类的成员ID,和方法ID一直都有效指导类被回收,除了很少情况下,类对象只有没在关联到类加载器的时候才会被垃圾回收机制回收,但是,在Android上是有可能出现的。备注,然而jclass是一个类引用,还有这类必须是protected权限或者有更高的访问权限(例如public)时才会被NewGlobalRel获得。

如果你想在一个类被第一次加载,和类被回收和重新加载的时候,自动缓存和重新缓存他们的ID(MethodID和fieldID),正确的初始化方式就是添加

一段像下面一样的代码在需要被访问的类里面

/*

* We use a class initializer to allow the native code to cache some

* field offsets. This native function looks up and caches interesting

* class/field/method IDs. Throws on failure.

*/privatestaticnativevoidnativeInit();static{nativeInit();}

4.Local and Global References

传递给Native方法的参数以及绝大多数JNI函数的返回值都是局部引用,局部引用只在Native函数调用期间有效,当Native函数调用完毕后,局部引用将失效。如果想持有一个对象的全局引用,应使用NewGlobalRef,NewGlobalWeakRef, NewGlobalRef保证全局引用的全局有效性,直到显式调用了DeleteGlobalRef。一旦对某个对象调用了NewGlobalRef,其引用计数将增加,VM将不会对其执行垃圾回收,因此在必要的时刻,应该显式调用DeleteGlobalRef来释放被引用的对象,让其被VM回收。(这应用到Object所有子类,包括jclass,jstring,和jarray)

唯一拿到非局部引用的方式是通过NewGlobalRef或者NewWeakGlobalRef.

如果你想长时间持有一个引用,你必须用全局引用,NewGlobalReef 方法拿局部引用作为参数返回一个全局引用,全局引用一直有效直到你调用DeleteGlobalRef为止。

当要缓存一个从FindClass返回的jclass时,下面这段代码经常被用到

jclass localClass=env->FindClass("MyClass");jclass globalClass=reinterpret_cast(env->NewGlobalRef(localClass));

JNI中可以有多个引用指向同一个对象,如果需要测试两个引用是否指向的是同一个对象,应该使用IsSameObject方法,而不是“==”。

JNI数据类型中,jfieldID以及jmethodID不是引用类型,不能对这两个类型的数据使用NewGlobalRef;由GetStringUTFChars以及GetByteArrayElements返回的原始数据指针也不是Object类型,返回的指针可以在多个线程中使用,直到调用了对应的Release方法。

有一种特殊的情况你需要注意:如果你将一个Native线程通过AttachCurrentThread方法Attach到VM,则该Native线程中的Local Reference始终不会自动free掉(线程Detach时才会free掉Local Reference),因此这种情况下,你需要手动Delete所以你创建的Local Reference。通常,在一个Loop中创建的任何Local Reference也需要手动Delete,因为VM能够创建的Local Reference的数量是有限的。

5.UTF-8 and UTF-16 Strings

Java中Char类型与String类型使用UTF-16编码,而C代码中的Char以及String(字符数组)使用UTF-8编码。为解决这个问题JNI提供GetStringUTFChars函数用于返回UTF-8编码的String,但调用该方法将导致额外的内存分配和转换操作,执行速率不及GetStringChars。

当通过Get方法获取到String后不要忘记调用Release方法。String Get方法返回的C方式的指针jchar ,jbyte指向的是原始的数据而并非局部引用,因此在调用Release方法之前这些指针都是有效的,同时也以为这,在Native方法Return后VM不会自动释放指向的资源。

不要向NewStringUTF方法传递非UTF-8编码的数据。一个常见的错误是从文件或者网络读取数据然后直接传递给NewStringUTF。除非你非常清楚数据是ASCII格式的,否则你应当对其做适当的转换,NewStringUTF只接受UTF-8编码的字符串。

6.Exceptions

在异常已经发生的时候不能调用JNI方法,你的代码应该通过ExceptionCheck和ExceptionOccured方法判断是否有异常发生,如果异常发生你可以:返回、处理异常清除异常。在异常发生的时候可以调用的JNI方法如下:DeleteGlobalRef

DeleteLocalRef

leteWeakGlobalRef

ExceptionCheck

ExceptionClear

ExceptionDescribe

ExceptionOccurred

MonitorExit

PopLocalFrame

PushLocalFrame

Release

7.Extended Checking

VM会对JNI调用做一些额外的运行时检查,包括:数组:分配数组的长度为负值。

非法指针:传递非法的jarray/jclass/jobject/jstring指针到JNI函数。或者传递NULL指针给一个必须为非NULL的参数。

类名:传递非法的类名,正确的类名的写法应类似”java/lang/String”。

DirectByteBuffer:传递非法的参数到NewDirectByteBuffer。

异常:在JNI Exception发生时,不做任何处理继续调用JNI方法。

JNIEnv*s:在线程中错误的使用JNIEnv(每个线程都有不同的JNIEnv,不能公用)。

jfieldID,jmethodID:使用和传递非法的或者不匹配的ID。

引用:Delete引用的时候调用的Delete方法不匹配。

Release Mode:在调用Release方法的时候使用了错误了mode(0,JNI_COMMIT,JNI_ABORT)。

类型安全:Native方法返回值不匹配。

UFT-8:传递了无效的UTF-8字符串到JNI方法。

启动JNI检查的方法

Emulator 默认已打开

ROOT过的设备adb shell stop

adb shell setprop dalvik.vm.checkjni true

adb shell start1

2

31

2

3

常规设备

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值