在android项目中要实现一个需求

为了性能的要求只能用c代码来实现功能。

这样就牺牲了java跨平台性。

通过加载.so的方式,把用c实现的模块集成到app中。

android提供jni层,作为一个适配器。

可以在java层调用c接口,在jni层可以通过java提供的反射机制调用java接口和创建java对象。

最后需求完成了,自测也没问题,嘻嘻,自己也开心了一下,但是提交测试后,测试人员马上报了一个bug。

出现local reference table overflow (max=512)这样的一个错误。我去,尽然出现了崩溃。

google和百度了半天,才发现原来发生了jni层的内存泄露,导致了崩溃。

jni层到底出现了啥内存泄露????

从java代码进入jni层本地代码调用时,Dalvik就创建了一张local reference表来存储local reference,这张表

的表项数有最大值限制,一般最大为都是512个,local reference表只有当退出jni层代码的调用是才会清除掉。

当表项数超过最大值限制时,Dalvik就会抛出异常,导致了崩溃。

何为local reference????

他是在native代码层他是一个本地变量和java对象的引用。

JNIEXPORT jint JNICALL Java_com_print(JNIEnv *env, jobject obj){

  char data[] = "daffdasf";

jstring content = (*env)->NewStringUTF(env, data);

}

变量content在 函数Java_com_print中是一个本地变量并且是String对象一个引用。所以会在local reference table

中追加一个表项来指向java对象。

其实每次进入native代码都会存在一个全局指向local reference table起始位置的ptr变量。

而上面函数中的content只是代表一个在local reference table中的偏移,通过ptr + content偏移

从local reference table获取java对象的值。

什么原因会发生local reference table overflow?????

那就是在一个循环中不断的创建local reference,而没有调用DeleteLocalRef去销毁这个local reference,

从而导致local reference table中表项不断增加,最后超过最大值,抱出了异常,导致了崩溃。

举两个例子哈:

例子1.

JNIEXPORT jint JNICALL Java_com_example(JNIEnv *env, jobject obj){

  char data[] = "daffdasf";

  int i = 0;

  for(i = 0;i < 1000;i++){

    jstring content = (*env)->NewStringUTF(env, data);

  }

}

例子1代码会导致local reference table overflow

例子2.

int Java_com_example(char * data){

   JNIEnv *env = NULL;
   JavaVM * vm = NULL;

  vm = getVm();

  (*vm)->AttachCurrentThread(vm, &env, NULL);

     jstring content = (*env)->NewStringUTF(env, data);

}

void callExample(){

  int i = 0;

  char data [] = "dafdfdasfds";

  for(i = 0;i < 1000;i++){

    Java_com_example(data);

  }

}

例子2代码会导致local reference table overflow

大家写jni的代码时要防止jni层的内存泄露,要注意用本地语言的方式清除本地语言获得的内存,

也要注意local reference和global reference的使用。

当然jni官文文档中没有告诉我们jni层实现的细节,只告诉我们如何规范的编写jni代码,这当然是正确的做法。

对于使用者来说只需要关注的他的使用不需要关注实现细节,这样就保证了可扩展性。

所以对于不同的对local reference的实现可能结果不一样,就有可能不出现上面local reference table overflow的错误了。

android 内存泄露之jni local reference table overflow (max=512)的更多相关文章

  1. JNI内存泄露JNI ERROR (app bug): local reference table overflow (max=512)

    原因是没即时释放对象,原本的代码是这样 static jobject getMaps(JNIEnv *env,jclass obj) { jclass stringbuilder_class = (* ...

  2. JNI Local Reference Changes in ICS

    [This post is by Elliott Hughes, a Software Engineer on the Dalvik team. — Tim Bray] If you don’t wr ...

  3. (转)专项:Android 内存泄露实践分析

    今天看到一篇关于Android 内存泄露实践分析的文章,感觉不错,讲的还算详细,mark到这里. 原文发表于:Testerhome: 作者:ycwdaaaa ;  原文链接:https://teste ...

  4. [转]深入Android内存泄露

    深入内存泄露 Android应用的内存泄露,其实就是java虚拟机的堆内存泄漏. 当然,当应用有ndk,jni时,没有及时free,本地堆也会出现内存泄漏. 本文只是针对JVM内存泄漏应用,进行阐述分 ...

  5. Android内存泄露的原因

    (一)释放对象的引用,误将一个本来生命周期短的对象存放到一个生命周期相对较长的对象中,也称“对象游离“.隐蔽的内部类(Anonymous Inner Class): mHandler = new Ha ...

  6. Android 内存泄露总结(附内存检测工具)

    https://segmentfault.com/a/1190000006852540 主要是分三块: 静态储存区:编译时就分配好,在程序整个运行期间都存在.它主要存放静态数据和常量. 栈区:当方法执 ...

  7. 【android内存泄露】 WebView篇

    在咱的博客园app里,新闻的内容使用WebView展示的.在测试中,咱重复进入.退出某个新闻10多次,观察到 Objects一直在不断增长,反复触发GC,但是一直回收不了,占用的内存越来越高,于是警觉 ...

  8. Android内存泄露总结

    内存泄露是如何产生的? 当一个对象已经不需要再使用了,本该被回收时,而有另外一个正在使用的对象持有它的引用从而导致它不能被回收,这导致本该被回收的对象不能被回收而停留在堆内存中,这就产生了内存泄漏. ...

  9. Android内存泄露分析之StrictMode

    转载请注明地址:http://blog.csdn.NET/yincheng886337/article/details/50524709 StrictMode(严格模式)使用 StrictMode严格 ...

随机推荐

  1. 【Linux】vi(vim)起步学起来有些困难,一步一步温习

    以Tomcat的配置文件service.xml为例,记录.学习vi的最常用操作. > 什么是vi or vim? [nicchagil@localhost bak]$ man vi VIM() ...

  2. 回车和换行在linux下和windows下

    今天,我总算搞清楚"回车"(carriage return)和"换行"(line feed)这两个概念的来历和区别了. 1.  在计算机还没有出现之前,有一种叫 ...

  3. Codeoforces 558 B. Duff in Love

    //   B. Duff in Love time limit per test 2 seconds memory limit per test 256 megabytes input standar ...

  4. reason: -[UIKBBlurredKeyView candidateList]: unrecognized selector sent to instance

    reason: -[UIKBBlurredKeyView candidateList]: unrecognized selector sent to instance 发现上线的app一直会有这个cr ...

  5. Tar命令用法详解

    tar -c: 建立压缩档案-x:解压-t:查看内容-r:向压缩归档文件末尾追加文件-u:更新原压缩包中的文件 这五个是独立的命令,压缩解压都要用到其中一个,可以和别的命令连用但只能用其中一个.下面的 ...

  6. 关于两个php.ini的说明

    关于两个php.ini的说明     Apache 和 php下各有一个php,ini文件 D:\wamp\bin\apache\apache2.4.9\bin\php.ini用于web访问时的配置文 ...

  7. Creating Excel File in Oracle Forms

    Below is the example to create an excel file in Oracle Forms.Pass the Sql query string to the below ...

  8. MyBatis Mapper 接口如何通过JDK动态代理来包装SqlSession 源码分析

    我们以往使用ibatis或者mybatis 都是以这种方式调用XML当中定义的CRUD标签来执行SQL 比如这样 <?xml version="1.0" encoding=& ...

  9. poj 3122 (二分查找)

    链接:http://poj.org/problem?id=3122 Pie Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 1 ...

  10. 堆内存指针的管理类(禁,引数(指针copy),值copy,移)

    //copyp template<typename T> class pm_copyP{ public: pm_copyP(T* _p); pm_copyP(const pm_copyP& ...