java.lang.OutOfMemoryError:位图大小超出 VM 预算

Posted

技术标签:

【中文标题】java.lang.OutOfMemoryError:位图大小超出 VM 预算【英文标题】:java.lang.OutOfMemoryError: bitmap size exceeds VM budget 【发布时间】:2011-06-04 10:51:40 【问题描述】:

我正在尝试将应用程序的布局从纵向更改为横向,反之亦然。 但是,如果我经常或不止一次这样做,那么有时我的应用程序会崩溃。下面是错误日志。请建议可以做什么?

 01-06 09:52:27.787: ERROR/dalvikvm-heap(17473): 1550532-byte external allocation too large for this process. 01-06 09:52:27.787: ERROR/dalvikvm(17473): Out of memory: Heap Size=6471KB, Allocated=4075KB, Bitmap Size=9564KB 
 01-06 09:52:27.787: ERROR/(17473): VM won't let us allocate 1550532 bytes 
 01-06 09:52:27.798: DEBUG/skia(17473): --- decoder->decode returned false
 01-06 09:52:27.798: DEBUG/androidRuntime(17473): Shutting down VM
 01-06 09:52:27.798: WARN/dalvikvm(17473): threadid=3: thread exiting with uncaught exception (group=0x4001e390)
 01-06 09:52:27.807: ERROR/AndroidRuntime(17473): Uncaught handler: thread main exiting due to uncaught exception
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473): java.lang.RuntimeException: Unable to start activity ComponentInfo: android.view.InflateException: Binary XML file line #2: Error inflating class <unknown
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2596) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2621) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at android.app.ActivityThread.handleRelaunchActivity(ActivityThread.java:3812) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at android.app.ActivityThread.access$2300(ActivityThread.java:126) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1936) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at android.os.Handler.dispatchMessage(Handler.java:99) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at android.os.Looper.loop(Looper.java:123) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at android.app.ActivityThread.main(ActivityThread.java:4595) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at java.lang.reflect.Method.invokeNative(Native Method)
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at java.lang.reflect.Method.invoke(Method.java:521) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:860) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:618) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at dalvik.system.NativeStart.main(Native Method)
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473): Caused by: android.view.InflateException: Binary XML file line #2: Error inflating class <unknown
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at android.view.LayoutInflater.createView(LayoutInflater.java:513) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at com.android.internal.policy.impl.PhoneLayoutInflater.onCreateView(PhoneLayoutInflater.java:56) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at android.view.LayoutInflater.createViewFromTag(LayoutInflater.java:563) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at android.view.LayoutInflater.inflate(LayoutInflater.java:385) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at android.view.LayoutInflater.inflate(LayoutInflater.java:320) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at android.view.LayoutInflater.inflate(LayoutInflater.java:276) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at com.android.internal.policy.impl.PhoneWindow.setContentView(PhoneWindow.java:207) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at android.app.Activity.setContentView(Activity.java:1629) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at onCreate(Game.java:98)
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1047) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2544) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     ...  12 more
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473): Caused by: java.lang.reflect.InvocationTargetException 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at android.widget.LinearLayout.<init>(LinearLayout.java:92) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at java.lang.reflect.Constructor.constructNative(Native Method)
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at java.lang.reflect.Constructor.newInstance(Constructor.java:446) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at android.view.LayoutInflater.createView(LayoutInflater.java:500) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     ...  22 more
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473): Caused by: java.lang.OutOfMemoryError: bitmap size exceeds VM budget
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at android.graphics.BitmapFactory.nativeDecodeAsset(Native Method)
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at android.graphics.BitmapFactory.decodeStream(BitmapFactory.java:464) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at android.graphics.BitmapFactory.decodeResourceStream(BitmapFactory.java:340) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at android.graphics.drawable.Drawable.createFromResourceStream(Drawable.java:697) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at android.content.res.Resources.loadDrawable(Resources.java:1705) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at android.content.res.TypedArray.getDrawable(TypedArray.java:548) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at android.view.View.<init>(View.java:1850) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at android.view.View.<init>(View.java:1799) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     at android.view.ViewGroup.<init>(ViewGroup.java:296) 
 01-06 09:52:27.857: ERROR/AndroidRuntime(17473):     ...  26 more 

【问题讨论】:

下次使用“格式代码”按钮,而不是“块引用”按钮。块引用按钮弄乱了消息 我也收到此错误,但没有任何解决方案。当我尝试从 MetaDataRetriver 类获取视频文件帧时出现此错误,我的代码 MediaMetadataRetriever mRetriever = new MediaMetadataRetriever(); mRetriever.setDataSource(this.videoPath); for (int i = 0; i java.lang.OutOfMemoryError: bitmap size exceeds VM budget - Android 的可能重复项 【参考方案1】:

将这行代码添加到“应用程序”标签内的 Android 清单文件中:- android:largeHeap="true"

<application
         android:theme="@android:style/Theme.NoTitleBar"
        android:allowBackup="true"
        android:largeHeap="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
       >

【讨论】:

【参考方案2】:

您可以使用opts.inSampleSize=2; or opts.inSampleSize=4 解决问题

BitmapFactory.Options opts = new BitmapFactory.Options();
// opts.inJustDecodeBounds = true; 
opts.inSampleSize=2;    
Bitmap myBitmap = BitmapFactory.decodeFile(st_imagepath,opts);

【讨论】:

【参考方案3】:

我对@hp.android 的出色回答做了几个扩展:

    我没有在每个 XML 类中定义 RootView ID,而是使用 android.R.id.content 动态查找根视图。 正如@Mario Lenci 所说,AdapterView 不支持removeAllViews,所以我在调用removeAllViews() 之前检查了视图的类型。 IMO 在每次 onDestroy 调用中调用 System.gc() 不是一个好主意(例如 this answer),因此我已将其删除。 我创建了一个静态 onDestroy 方法,我在所有自定义 Activity 类中从 onDestroy 调用该方法。 我添加了一些额外的错误保护和日志记录。

代码如下:

    public static void onDestroy(Activity activity) 
        View rootView = null;

        try             
            rootView = ((ViewGroup) activity.findViewById(android.R.id.content))
                    .getChildAt(0);
         catch (Exception e) 
            Log.w("Cannot find root view to call unbindDrawables on",
                  activity);
        

        if (rootView != null) 
            Log.d("unbindDrawables", activity, rootView);
            unbindDrawables(rootView);
        
    

    /**
      * Utility method to unbind drawables when an activity is destroyed.  This
      * ensures the drawables can be garbage collected.
      */
     public static void unbindDrawables(View view) 
         if (view.getBackground() != null) 
             view.getBackground().setCallback(null);
         

         if (view instanceof ViewGroup) 
             for (int i = 0; i < ((ViewGroup) view).getChildCount(); i++) 
                 unbindDrawables(((ViewGroup) view).getChildAt(i));
             

             try                 
                 // AdapterView objects do not support the removeAllViews method
                 if (!(view instanceof AdapterView))              
                     ((ViewGroup) view).removeAllViews();
                 
              catch (Exception e) 
                 Log.w("Ignore Exception in unbindDrawables", e);
             
         
     

【讨论】:

【参考方案4】:

首先通过在您认为会产生此错误的地方捕获 OutOfMemory 错误来阻止您的应用程序在运行时崩溃,如下所示:

try 

...


catch(OutOfMemoryError error)  
    //decide what to do when there is not more memory available

【讨论】:

【参考方案5】:

我多次收到 java.lang.OutOfMemoryError: Bitmap Size Exceeds VM Budget 错误

1) 一段时间后从应用程序切换并返回应用程序时(按主页按钮并浏览一些网址后)

2) 频繁登录/注销应用时(10秒内)

3) 水平/垂直切换设备时

最后我通过以下方式解决了错误

    public void clearAllResources() 

    // Set related variables null

    System.gc();
    Runtime.getRuntime().gc();
   

【讨论】:

正如 Android 性能文档所述,您不应该尝试强制 GC,您应该让 Davlik 自己处理它。充其量这隐藏了内存问题的原因并替代了 CPU 性能损失。 @Rafiq,您能否解释一下何时调用此函数 clearAllResources()?【参考方案6】:

这是一个实用的答案,我在运行时试图避免这个问题

Runtime.getRuntime().gc();

调用垃圾收集器是个好主意。

【讨论】:

【参考方案7】:

我发现开发 Android 应用程序最常见的错误之一是

java.lang.OutOfMemoryError:位图大小超出 VM 预算

错误。我在更改方向后使用大量位图的活动上经常发现此错误:活动被销毁,再次创建,布局从 XML 中“膨胀”,消耗了可用于位图的 VM 内存。

前一个 Activity 布局上的位图没有被垃圾收集器正确释放,因为它们交叉引用了它们的 Activity。经过多次实验,我找到了一个非常好的解决这个问题的方法。

首先,在 XML 布局的父视图上设置“id”属性:

    <?xml version="1.0" encoding="utf-8"?>
    <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
        android:layout_
        android:layout_
        android:id="@+id/RootView">
        ...

然后,在 Activity 的 onDestroy() 方法上,调用 unbindDrawables() 方法传递对父 View 的引用,然后执行 System.gc()

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

        unbindDrawables(findViewById(R.id.RootView));
        System.gc();
    

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

这个unbindDrawables() 方法递归地探索视图树并且:

    移除所有背景可绘制对象的回调 删除每个视图组中的子项

【讨论】:

这解决了我的问题...总是得到一个 layoutinflate 异常,在一段时间后为带有许多图形的大型布局充气...谢谢! 伟大的代码......我遇到的唯一错误是布局中有adapterView。它们在 removeAllViews() 上导致 UnsupportedOperationException,只是在该行添加了一个 try catch,一切正常。 如何处理用setTheme方法添加的背景图片?? 还是没有运气:(【参考方案8】:

http://yekmer.posterous.com/android-memory-management

可能对这个网站有帮助。你应该释放所有参考。

【讨论】:

【参考方案9】:

Angel,这里的 *** 上也有类似的问题@OutOfMemory exception when loading bitmap from external storage

从那里我推断 SoftReference 也可能是解决方案的替代方案。请检查帖子,它可能会有所帮助。

【讨论】:

【参考方案10】:

图片来源是哪里来的?您的 drawables 文件夹中有 9MB 的图像吗?

关于 Android 的原生位图内存管理问题的讨论很多,例如http://code.google.com/p/android/issues/detail?id=10821、http://code.google.com/p/android/issues/detail?id=8488、http://groups.google.com/group/android-developers/browse_thread/thread/ed57849ef705d421/11af8362d77a8cf4?lnk=raot。每次我完成位图时,我都非常幸运地显式调用了 recycle() 和 System.gc(),例如:

private void changeImage() 
     if (mCurrentImage != null) 
          mCurrentImage.recycle();
          mCurrentImage = null;
          System.gc();
     
     mCurrentImage = getNewImage();

【讨论】:

【参考方案11】:

1550532 字节的分配听起来很大。我猜你的布局引用了某种位图图像,也许是 PNG?这个PNG的尺寸是多少?如果它真的很大并且您有多个它们,那么您可能没有足够的内存来一次加载它们。

此外,如果这是由旋转触发的,请记住您的 Activity 已被销毁并重新创建,因此您可能需要小心静态引用。看看这个Google blog post。您需要注意在代码中保留对可能导致泄漏的事物的静态引用

【讨论】:

【参考方案12】:

您正在泄漏资源(很可能是Drawable 对象)。试试What Android tools and methods work best to find memory/resource leaks?中的工具

【讨论】:

以上是关于java.lang.OutOfMemoryError:位图大小超出 VM 预算的主要内容,如果未能解决你的问题,请参考以下文章