【问题标题】:I keep getting a java.lang.outofmemory error [duplicate]我不断收到 java.lang.outofmemory 错误 [重复]
【发布时间】:2013-08-10 15:05:35
【问题描述】:

我正在尝试为我的应用设置一个背景,该背景可以很好地适用于平板电脑和手机。所以我得到了一个尺寸为 2560 x 1600 但只有 44k 大小的 png 背景。然而,每次我运行应用程序时,我都会收到这个。

08-08 14:50:21.251: E/AndroidRuntime(2924): FATAL EXCEPTION: main
08-08 14:50:21.251: E/AndroidRuntime(2924): java.lang.RuntimeException: Unable to start activity ComponentInfo{com.reflap.reflap/com.reflap.reflap.MainActivity}: android.view.InflateException: Binary XML file line #7: Error inflating class <unknown>
08-08 14:50:21.251: E/AndroidRuntime(2924):     at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2211)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2261)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at android.app.ActivityThread.access$600(ActivityThread.java:141)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1256)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at android.os.Handler.dispatchMessage(Handler.java:99)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at android.os.Looper.loop(Looper.java:137)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at android.app.ActivityThread.main(ActivityThread.java:5103)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at java.lang.reflect.Method.invokeNative(Native Method)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at java.lang.reflect.Method.invoke(Method.java:525)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:737)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:553)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at dalvik.system.NativeStart.main(Native Method)
08-08 14:50:21.251: E/AndroidRuntime(2924): Caused by: android.view.InflateException: Binary XML file line #7: Error inflating class <unknown>
08-08 14:50:21.251: E/AndroidRuntime(2924):     at android.view.LayoutInflater.createView(LayoutInflater.java:620)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at com.android.internal.policy.impl.PhoneLayoutInflater.onCreateView(PhoneLayoutInflater.java:56)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at android.view.LayoutInflater.onCreateView(LayoutInflater.java:669)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at android.view.LayoutInflater.createViewFromTag(LayoutInflater.java:694)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at android.view.LayoutInflater.rInflate(LayoutInflater.java:755)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at android.view.LayoutInflater.inflate(LayoutInflater.java:492)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at android.view.LayoutInflater.inflate(LayoutInflater.java:397)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at android.view.LayoutInflater.inflate(LayoutInflater.java:353)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at com.android.internal.policy.impl.PhoneWindow.setContentView(PhoneWindow.java:267)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at android.app.Activity.setContentView(Activity.java:1895)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at com.reflap.reflap.MainActivity.onCreate(MainActivity.java:18)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at android.app.Activity.performCreate(Activity.java:5133)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1087)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2175)
08-08 14:50:21.251: E/AndroidRuntime(2924):     ... 11 more
08-08 14:50:21.251: E/AndroidRuntime(2924): Caused by: java.lang.reflect.InvocationTargetException
08-08 14:50:21.251: E/AndroidRuntime(2924):     at java.lang.reflect.Constructor.constructNative(Native Method)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at java.lang.reflect.Constructor.newInstance(Constructor.java:417)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at android.view.LayoutInflater.createView(LayoutInflater.java:594)
08-08 14:50:21.251: E/AndroidRuntime(2924):     ... 24 more
08-08 14:50:21.251: E/AndroidRuntime(2924): Caused by: java.lang.OutOfMemoryError
08-08 14:50:21.251: E/AndroidRuntime(2924):     at android.graphics.BitmapFactory.nativeDecodeAsset(Native Method)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at android.graphics.BitmapFactory.decodeStream(BitmapFactory.java:503)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at android.graphics.BitmapFactory.decodeResourceStream(BitmapFactory.java:356)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at android.graphics.drawable.Drawable.createFromResourceStream(Drawable.java:800)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at android.content.res.Resources.loadDrawable(Resources.java:2105)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at android.content.res.TypedArray.getDrawable(TypedArray.java:601)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at android.widget.ImageView.<init>(ImageView.java:127)
08-08 14:50:21.251: E/AndroidRuntime(2924):     at android.widget.ImageView.<init>(ImageView.java:117)
08-08 14:50:21.251: E/AndroidRuntime(2924):     ... 27 more

我不明白这么小的尺寸怎么会出现内存不足的情况。

这是我的 imageview 布局 xml

<ImageView
        android:id="@+id/imageView1"
        android:layout_width="wrap_content"
        android:layout_height="fill_parent"
        android:layout_alignParentLeft="true"
        android:layout_alignParentTop="true"
        android:scaleType="center"
        android:src="@drawable/back" />

【问题讨论】:

标签: android image android-layout


【解决方案1】:

44k 是压缩后的大小。为了将其用作背景,它被解压缩为位图。位图的内存大小是每个像素的颜色格式大小乘以宽度乘以高度。更糟糕的是,如果它被缩放,系统也将需要更多的内存来进行缩放。

因此,例如,如果您使用这种颜色格式: http://developer.android.com/reference/android/graphics/Bitmap.Config.html#RGB_565

那么位图所需的内存为 2 x 2560 x 1600 = 7.8125 MB。

有一个获取更多内存的清单选项: http://developer.android.com/guide/topics/manifest/application-element.html#largeHeap

尽管您还应该确保为较低 DPI 的设备提供较小的资源,并且只读取您需要的尽可能多的数据。如果您不需要每个像素,BitmapFactory 类允许您在读取数据时跳过像素。 BitmapRegionDecoder 让您可以只读取您需要的图像部分。

【讨论】:

  • 大多数情况下不需要使用 largeHeap。堆越大,垃圾收集越频繁,暂停时间越频繁
【解决方案2】:

在内部,您的 .png 以每个像素的整数表示(默认情况下)。一个整数是 4 个字节。如果你做数学:

2560 pixels * 1600 pixels * 4 bytes / pixel = 16,384,000 bytes = 15.625 MB

旧版 Android 设备上每个应用的内存限制为 16 MB。 Portable Network Graphics 格式通常是经过压缩的,这就是为什么它的文件大小和在内存中表示图像的大小存在明显差异的原因。此外,即使您的图像没有或不需要 Alpha 通道,Android 也会在 Alpha 通道上浪费一个字节。

【讨论】:

【解决方案3】:

避免使用位图填充背景。它们不会给出最好的结果,并且会在不保留屏幕比例的情况下进行缩放。

使用 9-patches、形状可绘制对象、简单的颜色和/或锚定到某个角落的小位图。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2019-05-05
    • 1970-01-01
    • 2013-10-03
    • 2021-12-10
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多