Android 4.X 系统加载 so 失败的原因分析

1 so 加载过程

so 加载的过程可以参考小米的系统工程师的文章loadLibrary动态库加载过程分析

2 问题分析

2.1 问题

年前项目里新加了一个 so库,但发现native 方法的找不到的 crash 好多,好些都是报了java.lang.unsatisfiedlinkerror native method not found,而且基本上是出现在4.x的系统里,特别是 4.4,4.2的系统。在网络上搜索相关的可能导致到这个问题的原因:

  • so 文件没有在对应架构的目录里找到;
  • 方法名有错误;

    2.2 分析1

    我们最开始是怀疑应用在安装时没有正确解压出对应的so文件到相应目录,因此加了相应统计来看发生crash的手机是否是因为找不到对应的 so文件导致的;但统计数据发现这些手机里都可以找到对应架构的 so文件,因此就排除了不存在so文件导致的 crash;

2.3 分析2

我们同事以前有发现在Android 4.x系统里,如果so 文件是在应用启动时加载的,但使用时机却在后面的时间点,so加载进手机的内存可能会被系统由于资源紧张而回收掉,这种情况下,可以通过重新加载一次 so文件来减少相关的 crash,这种方法 fix了某个量很高的 so 相关的crash。但我们的 socrash 明显是不属于这种情况的,因为我们是通过 System.load() 方法加载完 so文件后,就调用相关的方法,这时内存肯定是还在的。在分析了一系列可能的原因后,怀疑这个crash 是因为应用安装时解压出来的 so文件是损坏的,因此我们尝试在第一次发生这个crash时,将这个crash catch住,然后在 catch块将原来目录下的 so文件删除掉,并重新从应用的安装目录解压出对应的so文件放到原来的目录,并加了相关的统计来验证。so的加载用了 Relinkder。相关的简化版本代码如下:

relinker.loadLibrary(getApplicationContext(), "so_name");
try{
    // call native method
} catch(UnsatisfiedLinkError e) {
    //some stats
     String library = "so_name";
     String libName = System.mapLibraryName(library);
     File workaroundLibDir = getApplicationContext().getDir("lib", Context.MODE_PRIVATE);
     File workaroundLibFile = new File(workaroundLibDir, libName);
     workaroundLibFile.delete();

     apkLibraryInstaller.installLibrary(
                            getApplicationContext()
                            , supportedAbis()
                            , libName
                            , workaroundLibFile
                            , relinker
                    );
     System.load(workaroundLibFile.getAbsolutePath());
     //call native method
     // some stats
}

2.4 分析3

在使用的 2.3 的解决方法后,我们的 sojava.lang.unsatisfiedlinkerror native method not found 大部分消失了。理论上使用过一次重新解压so 文件后,这个用户在下一次升级前都应该不会再发生了类似的 crash了,但我们的统计数据发现,有些用户每一次启动都需要进入catch块来避免crash,而每次都可以通过 reload来正常使用我们的应用,这至今还是个迷,还没有想明白是什么情况会导致这个问题?手机的存储有问题?但其他的so又没有这个问题。希望如果有同行解决过类似的问题的,指点一下。

3 总结

Android 4.X 系统加载 so 后,出现 java.lang.unsatisfiedlinkerror native method not foundcrash的原因除了网上所说的 不存在这个so 和 方法名有问题(商用的应用应该不会有这个问题的)外,还有两个原因:

  • so 加载进系统的内存被系统由于资紧张而回收了,这种情况下直接再load一下 so 文件就可以解决大部分;
  • so 文件有问题,这种情况下,可以通过重新从应用安装目录解压出对应的 so 文件并重新加载来解决大部分;

这两种方法不能保证可以100%解决问题,但可以减少大部分问题(90%);

原文地址:https://www.cnblogs.com/WoodJim/p/8468865.html

时间: 2024-10-01 04:37:59

Android 4.X 系统加载 so 失败的原因分析的相关文章

(十) ng-inlude指令加载页面失败的原因和解决方法

angularjs中提供的ng-include指令,很类似于JSP中的<jsp:include>用来将多个子页面合并到同一个父页面中,避免父页面过大,可读性差,不好维护. 父页面parent.html代码如下: <html> <head> <script src="angular-1.2.2/angular.js"></script> <script> function rootController($scope,

React Native 断点调试 跨域资源加载出错问题的原因分析

写在前面 ----如果从头开始看还没解决,试试文章最后的绝招 闲来无事,折腾了一下React Native,相比之前,开发体验好了不少.但在真机断点调试那里遇到了跨域资源加载出错的问题,一番探索总算解决,目测是RN新版本调试服务的bug. 遇到类似问题的同学应该不少,这里做下记录,有需要的可以参考下. 如何断点调试 首先,在真机上加载运行RN应用(过程略). 然后,摇动手机,弹出开发菜单,选择"Debug JS Remotely". chrome会自动打开调试界面,地址是 http:/

记:Ubuntu14.04 Android加载项目失败

Android 加载项目失败: sdk/build-tools/android-4.4.2/aapt: error while loading shared libraries: libz.so.1: cannot open shared object file: No such file or directory 安装: sudo apt-get install lib32z1 记:Ubuntu14.04 Android加载项目失败,布布扣,bubuko.com

ubuntu系统服务器更换相同型号的主板后登录时提示加载文件系统失败

ubuntu系统服务器更换相同型号的主板再次登录时提示加载文件系统失败,且还有上次加载成功时间为xxxx-xxxx-xx,后经查找原因,主板系统时间太老(修改为当前日期时间即可),系统启动,文件加载与系统时间有关(貌似),有时间查找资料学习学习 注: SATA硬盘几种模式: IDE 是为了兼容性,将sata模拟成ide模式(比较常用) raid是磁盘阵列模式,一般人不用.(一般需要安装多个硬盘才能实现) ahci是真正的sata模式.(要想发挥硬盘性能,建议用此模式) ubuntu系统服务器更换

Android中的Glide加载图片

注意:在Android Studio的项目的build.gradle中添加: compile 'com.github.bumptech.glide:glide:3.6.1' 然后同步一下 目录: 使用Glide结合列表的样式进行图片加载 如果使用的是RecyclerView,可以在Adapter的onBindViewHolder方法中使用 当加载网络图片时,由于加载过程中图片未能及时显示,此时可能需要设置等待时的图片,通过placeHolder()方法 当加载图片失败时,通过error(Draw

Android下将图片加载到内存中

Android的系统的标准默认每个应用程序分配的内存是16M.所以来说是非常宝贵的,在创建应用的时候要尽可能的去节省内存,但是在加载一些大的文件的时候,比如图片是相当耗内存的,一个1.3M的图片,分辨率是2560X1920(宽X高)图片当加载到手机内存的时候就会请求19M的一块内存,这是远远超出了系统自带的内存空间,这时候应用程序就会挂掉,所以我们要进行图片的缩放处理,下面我就来带大家创建一个用来图片缩放的应用: 应用效果图如下: 核心代码的实现: package com.examp.loadp

Android运行时ART加载类和方法的过程分析

在前一篇文章中,我们通过分析OAT文件的加载过程,认识了OAT文件的格式,其中包含了原始的DEX文件.既然ART运行时执行的都是翻译DEX字节码后得到的本地机器指令了,为什么还需要在OAT文件中包含DEX文件,并且将它加载到内存去呢?这是因为ART运行时提供了Java虚拟机接口,而要实现Java虚拟机接口不得不依赖于DEX文件.本文就通过分析ART运行时加载类及其方法的过程来理解DEX文件的作用. 老罗的新浪微博:http://weibo.com/shengyangluo,欢迎关注! 在前面An

Android利用Volley异步加载数据完整详细示例(二)

MainActivity如下: package cc.y; import android.app.Activity; import android.content.Context; import android.graphics.Bitmap; import android.graphics.Bitmap.Config; import android.os.Bundle; import android.util.LruCache; import android.widget.ImageView;

Android中的动态加载机制

在目前的软硬件环境下,Native App与Web App在用户体验上有着明显的优势,但在实际项目中有些会因为业务的频繁变更而频繁的升级客户端,造成较差的用户体验,而这也恰恰是Web App的优势.本文对网上Android动态加载jar的资料进行梳理和实践在这里与大家一起分享,试图改善频繁升级这一弊病. Android应用开发在一般情况下,常规的开发方式和代码架构就能满足我们的普通需求.但是有些特殊问题,常常引发我们进一步的沉思.我们从沉思中产生顿悟,从而产生新的技术形式.如何开发一个可以自定义