1.5.2 一个类的内存消耗
首先,如果我们在代码中要使用一个类,例如以下代码:
Foo f = new Foo();
虚拟机在执行到这步时会做什么呢?
第一步是loadClass操作,将类信息从dex文件加载进内存:
1)读取.dex mmap中class对应的数据。
2)分配native-heap和dalvik-heap内存创建class对象。
3)分配dalvik-LinearAlloc存放class数据。
4)分配dalvik-aux-structure存放class数据。
第二步是new instance操作,创建对象实例:
1)执行.dex mmap中<clinit>和<init>的代码。
2)分配dalvik-heap创建class对象实例。
在这个过程中,可能还会分配dalvik-bitmap和jit-code-cache内存。如果class Foo引用了其他类型,那就还需要先按照同样的逻辑创建被引用的class。由此可见,在创建一个类实例的每一步都需要消耗内存。我们接下来大概计算一下new操作需要消耗的内存。
根据Dalvik虚拟机的代码,能够得知class根据类成员和函数的数目分配LinearAlloc和aux-structure的多少,以及class本身及函数需要的字节数。我们再根据一个应用中所有class的总量进行平均计算,得到以下一组数据。
第一步是loadClass操作,加载类信息:
.dex mmap(class def + class data): 载入一个类需要先读取259字节的mmap。
dalvik-LinearAlloc: 在LinearAlloc区域分配437字节,存放类静态数据。
dalvik-aux-structure: 在aux区域分配88字节,存放各种指针。
第二步是new instance操作,创建对象实例:
.dex mmap(code):为了执行类构造函数,还需要读取252字节的mmap。
dalvik-heap: 根据类的具体内容而变化。
可见在创建对象实例的操作中,Dalvik Other和.Dex Mmap部分就各需要约500字节的内存空间。但是考虑到4KB页面的问题,由于这些内存并不是连续分布的,所以可能需要分配多个4KB页面。当然由于很多类会在一起使用,使得实际的页面值不会那么多。
以我们举例的应用为例,总共有7042个类,启动后载入了1522个类,这时侯应用的.dex mmap内存消耗大约是5MB,平均后约为3.4KB。Dalvik Other的部分会少一些,但依然是远远超出需要使用的大小。