Android了解堆大小

时间:2012-05-20 22:45:23

标签: android heap virtual-machine dalvik ddms

我是Android开发的新手,我似乎无法掌握Java Out of Memory异常。我知道这意味着我的应用程序已经超过了VM的预算,但经过Google搜索这么多次后,我似乎仍然没有掌握这个概念。我担心我的应用程序会占用太多内存,因为每个屏幕有六个按钮选择器,每个选择器有两个位图,根据属性选项卡每个大约20 kb。在我的根目录G2x上,我已将VM预算设置为12mb,重新启动了我的手机并运行了我的应用程序,没有任何问题。我在每个onDestroy()上取消绑定drawable,并暗示GC也在这里运行。在模拟器中使用应用程序一段时间后,在我的DDMS屏幕上单击“原因GC”,结果如下 ID = 1,堆大小6.133 MB,分配2.895MB,免费3.238 MB,%使用47.20,#对象52,623。

这是我不明白发生了什么的地方,我的模拟器被设置为24MB的VM。那个数字在哪里?我遇到的实际问题是,如果我将模拟器设置为16MB的VM,我的应用程序会因为内存不足异常而在第二个活动上崩溃。为什么我的手机上的VM设置为12 MB,或者我的旧HTC Magic手机上有12 MB的VM存量?你们还认为我的应用程序占用了太多内存吗?我不知道这些DDMS号码是否好。谢谢你的时间。

至于我的代码,我有XML布局中指定的每个图像,除了向它们添加侦听器之外,我不会以编程方式对它们执行任何操作。我在这里找到了这段代码,并将其添加到我拥有的每个活动中......

@Override
protected void onDestroy() {
    super.onDestroy();

    unbindDrawables(findViewById(R.id.myRootLayout));
    System.gc();
}

private void unbindDrawables(View view) {
    if (view.getBackground() != null) {
        view.getBackground().setCallback(null);
    }
    if (view instanceof ViewGroup && !(view instanceof AdapterView)) {
        for (int i = 0; i < ((ViewGroup) view).getChildCount(); i++) {
            unbindDrawables(((ViewGroup) view).getChildAt(i));
        }
        ((ViewGroup) view).removeAllViews();
    }
}

否则我所做的就是将onClickListeners添加到具有PNG背景的按钮。我想学习如何以编程方式指定按钮背景,但我需要选择功能,如焦点,按下,非聚焦但按下等,以使按钮背景根据用户交互而变化。我已经回顾了关于这个的文档,但它看起来势不可挡,这就是为什么我认为我从这里开始管理堆的基础知识,并努力在代码中指定选择器。这可能没有意义但是有一个“健康”的内存分配量,应用程序可以分配而不会接近内存不足异常?例如,如果一个应用程序分配6MB应该没问题,但8MB会推动它,在内存分配方面有这样的界限吗?再次感谢Alex Lockwood的回复我将再次阅读并重新阅读,直到这些内容对我有意义

2 个答案:

答案 0 :(得分:51)

在仿真器/设备上设置VM预算时,您正在做的是告诉堆允许的最大大小。在运行时,当Dalvik VM从操作系统请求系统内存时,堆的大小会动态增长。 Dalvik VM通常首先分配一个相对较小的堆。然后在每次GC运行后,它会检查有多少可用堆内存。如果可用堆与总堆的比率太小,则Dalvik VM将向堆中添加更多内存(最大配置堆大小)。

话虽这么说,你在DDMS屏幕上没有看到“24 mb”的原因是因为堆没有增长到它的最大大小。这使得Android可以充分利用手持设备上已有的少量内存。

至于为什么你的应用程序在模拟器而不是你的手机上崩溃,这看起来很奇怪(你确定这些数字是正确的吗?)。但是,您应该记住,内存是动态管理的,并且总内存利用率是根据许多外部因素(执行垃圾收集的速度/频率等)确定的。

最后,由于我上面提到的原因,很难确定您的应用程序基于您在上面提供的单行信息管理内存的程度。我们真的需要看一些你的代码。但是,OutOfMemoryError绝对值得担心,所以我肯定会考虑你的应用程序的内存使用情况。您可以考虑的一件事是在运行时使用inSampleSize类调用BitmapFactory来对位图图像进行采样。这有助于减少加载可绘制位图所需的内存量。要么你或者你可以降低你的drawables的分辨率(虽然每个20 kb听起来不错)。

答案 1 :(得分:17)

即使你的应用程序没有达到“24mb”(因设备而异)堆限制,你可能仍然会崩溃,因为Android需要一段时间来增加应用程序的堆空间。

在我的情况下,我在很短的时间内制作并倾倒了几张图像。

我经常收到OutOfMemoryError

似乎Android的速度不足以为我的应用增加堆空间。

我相信我使用清单文件中的largeHeap设置解决了这个问题。 启用该设置后,Android每次增加堆时都会留下更多可用内存,从而最大限度地降低达到当前限制的可能性。

我不使用24mb限制,但这个largeHeap conf非常方便。

您只需在largeHeap="true"

的应用标记上设置AndroidManifest.xml即可
<application
    android:icon="@drawable/ic_launcher"
    android:label="@string/app_name"
    android:largeHeap="true"
    android:theme="@style/AppTheme" >

尽管如此,请确保在处理图像时要小心,例如@Alex Lockwood建议。