Java自学者论坛

 找回密码
 立即注册

手机号码,快捷登录

恭喜Java自学者论坛(https://www.javazxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,会员资料板块,购买链接:点击进入购买VIP会员

JAVA高级面试进阶训练营视频教程

Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程Go语言视频零基础入门到精通Java架构师3期(课件+源码)
Java开发全终端实战租房项目视频教程SpringBoot2.X入门到高级使用教程大数据培训第六期全套视频教程深度学习(CNN RNN GAN)算法原理Java亿级流量电商系统视频教程
互联网架构师视频教程年薪50万Spark2.0从入门到精通年薪50万!人工智能学习路线教程年薪50万大数据入门到精通学习路线年薪50万机器学习入门到精通教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程MySQL入门到精通教程
查看: 391|回复: 0

针对Android 5.0以下手机的MutilDex解决方案

[复制链接]
  • TA的每日心情
    奋斗
    2024-4-6 11:05
  • 签到天数: 748 天

    [LV.9]以坛为家II

    2034

    主题

    2092

    帖子

    70万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    705612
    发表于 2021-7-13 10:17:06 | 显示全部楼层 |阅读模式

    和大家分享下大型APP处理Multidex的方案。

    65K问题的来源:

    .Java文件编译生成字节码文件,然后打包生成.dex时会按照类,方法等进行分类,使用IndexMap结构处理。

     

    所有的方法都存放在short[] 里, 而short的大小正是65K。 

    method的数量远多于class的数量,method的数量会先到65K,class、field同样存在65k的大小限制。

    统计时是把系统的,第三方的加上自己的,方法数量的合,都放到这个数组里进行计算, frameworkMethodids + libraryMdthodids + mycodemethodids。

    目前存在的解决方案:

    坊间存在一种插件的解决方案:

    把部分app的功能分离出来做成独立的工程,并把这些功能打包生成apk,在把apk改成.png类型作为资源文件放到asset中,在中工程用到时加载这些“资源”并调用之中的方式,实现了减小主工程方法数、动态加载指定的类完成需要的功能。抽离出来功能作为插件减小了主工程方法数、也带来了新工程的维护成本、和插件工程不能混淆的安全成本、而且跨2个功能的代码维护会变得复杂、调试成本提高。

    顶尖的Google提供了另外的一种Multidex方案,简单高效:

    1. 编译配置文件build.gradle的 android{}集合中添加 multDexEnable true

    2. 编译配置文件build.gradle的dependencies{}集合中添加依赖库  developCompile android.support:multidex:1.0

    3. App启动时,oncreate中添加MultiDex.install(context);


    如果APP运行在ART虚拟机的设备上,首次启动app加载时间,目前小于100ms,不存在加载慢的问题。ART的优化堪称完美

    如果运行在老旧的dalvik虚拟机上,手机app加载时间大约5-20s,会导致ANR。


    解决ANR的方法: 

    总体思路是:把这个MultiDex.install(this)放到异步线程中加载,同时又要保证在用户使用功能之前,异步线程已经把非主dex加载完成。

    从编译到运行起来,发生什么了呢?

    1. 编译生成allclasses.jar 生成classes.dex, classes2.dex, .... 多个。

    安装阶段:把.dex 优化成classes.odex, 注意,只优化第一个dex文件。

    首次运行:

    1. dexpath BaseClassLoader.pathlist 类, 把所有的dex都加载。解决了为什么多个dex里的方法也能被正确的index到,不会导致no such method 的crash

    2. odex, 这个过程是非常耗时的,如果第二个dex是5-7M的话大约要加载4s, google这个默认方式时间太长,会导致ANR。

    这就需要完成2个事情, 1. 把启动需要加载的类放到主dex中, 2. 显示loading界面,保证用户使用之前把dex加载完成。

    主dex中的类: 

    把这个MultiDex.install(this)放到异步线程中加载,主dex中存放application启动的1直接引用类,所有可能的2入口类,加上3第三方的调用。扫面这些类的直接引用类。统统放到主dex中。

    首先在编译打包阶段,在build task中添加一层task, 添加Blacklist.xml自定义的黑名单, 这个名单中的方法对应的类需要添加到主dex中,保证主dex能够加载APP启动时需要的必要的类。

     

    balcklist中包含了需要放到主dex中的类的path,addtask的这个task,把blacklist中的类放到了dextask中生成想要的dex。

    如何确定哪些类需要放到blcaklist中:

    粒度到方法层面,从主activity加载开始,加载了class B的public x 和public y, 则把B的x 和 y方法中,仅x, y方法中import到的类进行分析, 进行回溯添加。

    那如何能让工作更高效,能让查找更有保证呢? 请看下篇,基于字节码文件分析.class文件。 

    异步加载界面的现实时机:

    splash界面,主界面,在收到事件之前进行监听显示load界面。直到异步加载完成所有dex。 

     

    其他问题:

    主dex不够65k方法,主dex会有多少方法? 生成dex时会一直塞直到塞满65K。

    JNI层的回调类,和用到反射的类是需要放到主dex中的,这些BI,库加载都是需要运行就work的。 

     

    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|小黑屋|Java自学者论坛 ( 声明:本站文章及资料整理自互联网,用于Java自学者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2024-5-20 11:53 , Processed in 0.061526 second(s), 29 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表