全部博文(166)
分类: IT业界
2011-10-23 23:00:55
程序能否运行的快速而高效,这在资源有限的移动终端设备上尤其显得重要。因为即使目前最强大的移动终端的计算能力也无法和市场上普通的桌面PC相抗衡。简单地说,有两个原则在资源受限的系统中必须遵守:
不做不必要的事情。
不分配不必要的内存。
在Android中,设计人员已经从UI、框架、基础平台等多个层次的多个方面进行了优化。但对于开发者而言,在开发过程中,仍然需要针对消耗资源比较多的设计进行规避或者优化。在接下来的内容中,将会针对Android已经进行的优化做简略的介绍。对开发者在开发过程中需要注意的地方进行一些说明。
一、资源读取
对资源的读取可以分为对资源文件的读取和对挂载的SD卡上的内容的读取。
1.资源文件
资源文件的优化读取,Android在编译时对描述UI 的XML文件进行了优化。
2.SD卡
Android的图片浏览器等多媒体应用可以加载整个SD卡内的所有图像,在加载前会把数据做成数据库,不用每次扫描,这大大加快了启动速度。事实上扫描操作是通过MediaScanner来实现的,目前支持的文件类型在MediaFile.java中定义。主要包括音频、MIDI、视频、图片、播放列表等。MediaScannerService服务的启动仅在收到"android.intent.action.BOOT_
COMPLETED"、"android.intent.action.MEDIA_MOUNTED"、"android.intent.action.MEDIA_
SCANNER_SCAN_FILE"后才会启动。
当然,在SD卡容量较大且文件较多时,MediaScannerService服务将会运行一段不短的时间,这对电池的持续能力会造成一定的影响,尤其是在电池技术始终不能有显著突破的前提下。
二、DEX文件和APK加载
在Android中,对编译出来的DEX字节码和APK文件的加载过程,也进行了尽可能的优化。
对于预置应用,Android会在系统编译后,生成优化文件,以ODEX后缀结尾,这样在发布时除APK文件(不包含DEX)外,还有一个相应的ODEX文件。
对于非预置应用,运行前,Android会优化DEX文件,在第一次启动应用时,执行文件的DEX被优化成DEY文件并放在/data/dalvik-cache目录。如果应用的APK文件不发生变化,DEX文件不会被重新生成,加快了以后的启动速度。APK文件的加载过程如图1所示。
图1 APK加载
DEX文件由header、string_ids、type_ids、proto_ids、field_ids、method_ids、class_defs、data等几部分构成。图2显示了这几部分内容在DEX文件中的布局。
图2 DEX格式
在Java中,每一个类会被编译成相应的CLASS文件,一个应用会定义若干个类,这就导致同一个应用的多个CLASS文件中会存在冗余信息,而在Android中,“dx”工具会将同一个应用的所有CLASS文件内容整合到一个DEX文件中,这样就减小了整体的文件尺寸,I/O操作也提高了类的查找速度。“dx”工具整合CLASS文件的过程如图3所示。
图3 “dx”工具整合CLASS文件的过程
原来每个CLASS文件中的常量池,在DEX文件中由一个常量池来统一管理,具体如图4所示。
图4 DEX的常量池
具体到DEX文件,经过“dx”工具优化后的内部逻辑如图5所示。
图5 DEX的内部逻辑
三、虚拟机和平台实现
在虚拟机和原生库层面,Android同样进行了很多的优化。
1.虚拟机
虚拟机中指令的解释时间主要分为3个方面:分发指令、访问运算数、执行运算。其中“分发指令”这个环节对性能的影响最大,为了加快运行速度,必须提高分发指令的速度。
与传统的Java虚拟机基于栈不同,Dalvik是基于寄存器的。基于寄存器的虚拟机实现,虽然在硬件通用性上稍逊一筹,但是数据处理速度却有明显的改善,可以更为有效地减小冗余指令的分发和减小内存的读写访问。
Dalvik虚拟机针对移动终端所做的优化,使得其不需要很快的CPU速度和大量的内存空间。根据Google的测算,Android的早期版本只需要64MB的RAM即可使系统正常运转,其中24MB被用于底层系统的初始化和启动,另外20MB被用于高层启动、高层服务。当然,随着Android版本的不断升级和应用功能的扩展,Android对内存的消耗也在逐渐增加。
另外需要注意的是,Dalvik并不是按照Java虚拟机的规范来实现的,两者并不兼容。
Java虚拟机运行的是Java字节码,而Dalik虚拟机运行的则是其专有的DEX(Dalvik Executable)字节码。
在Java SE程序中的Java类会被编译成一个或者多个字节码文件(.class),然后打包成JAR文件。在执行期间,Java虚拟机会从JAR文件抽取相应的CLASS文件并从中读取指令和数据。而Android虽然也是基于Java语言进行编程的,但是在编译成CLASS文件后,Android会通过“dx”工具将应用所有的CLASS文件转换一个DEX文件,接着将DEX和应用的其他如资源文件等一起打包构成APK文件,而后Dalvik虚拟机会从其中读取指令和数据。图6显示了Android的编译过程。
图6 Android的编译过程
Dalvik虚拟机的主要特征包括:专有的DEX字节码、支持新的操作码、文件结构非常简洁、使用等长的指令、借以提升解析速度、尽量扩大只读结构的大小、借以提高跨进程的数据共享比例。
2.平台优化
充分挖掘CPU的性能,针对armv5te进行了优化,充分利用armv5te的执行流水线来提高执行的效率。
3.C库
优化和裁剪的libc库Bionic,具有更高的效率、低内存占用、非常快和小的线程实现、内置了对Android特有服务的支持等特点。
4.创建进程
Linux在创建一个新进程时利用了写时拷贝(Copy-on-Write)机制,使得创建一个新的进程非常高效。Android中每个进程都是基于虚拟机的,并且也要加载基本的库,实际上这些都是可以共享的。基于这方面的考虑,Andoid引入了写时拷贝机制,使得Android启动一个新的进程,实际上并不消耗很多的内存和CPU资源。
另外,Android在后台一直有个Zygote虚拟机在运行,实际上是一个虚拟机实例的孵化器。如果要启动一个新的应用,Zygote就会创建出一个新的子进程来执行该应用程序,十分高效。图7显示了Zygote创建子进程的过程。
图7 Zygote创建子进程
Android在开机过程中,会首先启动Zygote虚拟机而不是系统服务器(System Server),也是出于利用写时拷贝机制创建进程比较高效的考虑。Zygote虚拟机在启动后会完成虚拟机的初始化、库的加载、预置类库的加载和初始化等操作。并在系统需要一个新的虚拟机对象时,Zygote可以通过写时拷贝机制高效地创建出新的虚拟机对象。
5.渲染机制
在进行渲染时,Android 会根据变化的部分进行局部更新,并不是每次都需要重绘整个屏幕。首先计算需要重绘的区域(mInvalidRegion),如果DisplayHardware:: UPDATE_ ON_DEMAND ,则通过设定需要重绘的区域的边界来进行局部重绘。
(摘自华清远见系列图书《Android多媒体编程从初学到精通》)