Android大图片引起的内存问题(OOM)

在Android应用开发中,对一个应用在内存使用大小是有限制。在应用中如果大量的使用Bitmap很有可能导致内存溢出的问题(OOM),出现这种问题后,第一感觉就是头疼,但是又不得不解决,在网上查了一些资料后,找到解决方法。


问题描述:

在一个项目中,要用Gallery来显示多张不同的图片,在给Gallery的每个Item设置图片显示时,想到Bitmap如果不手动的recycle,系统是不会回收其所在的资源的,因为Android为了提高效率,Bitmap真正的位图数据是在ndk中C写的。所以就直接使用setImageResource(iconResId); 但是在部分配置比较低的设备上还是出现OOM错误。


问题分析:

在setImageResource中,是根据iconResId的到一个drawable,在将drawable显示在view中。在查阅资料后得知:Android对于直接通过资源id载入的资源做了Cache,这样下次再需要次资源时直接从cache中得到。

在Gallery中用到多个大的图片,每个图片都会在cache中进行缓存,即使在Gallery中该View被回收了。这就是导致内存溢出的问题。


问题解决:

解决的一般办法:

一:在内存引用上做些处理,常用的有软引用、强化引用、弱引用

二:在内存中加载图片时直接在内存中做处理,如:边界压缩

三:动态回收内存

四:优化Dalvik虚拟机的堆内存分配

五:自定义堆内存大小


而对于前面我们找到的问题,我们对第一个和第二个解决办法进行整合就能够解决这个问题,(写代码):

在项目中主要将前两种方法进行了整合。

[java]  view plain  copy
  1. private static LinkedHashMap<String, Bitmap> hardManager;  
  2.     private static ConcurrentHashMap<String, SoftReference<Bitmap>> softManager;  

设置上面的两个变量,分别保存包名到Bitmap的强引用的映射和软引用的映射。

在通过包名获取bitmap时,首先在hardManager中判断是否包含该包名,如果有,直接取出;如果没有,则在softManager中判断,有取出;没有就通过下面方法获取。获取之后在插入到hardManager中。在插入时,做判断只保证hardManager的容量为10,如果大于10,则将最早插入的删除。

你可能感兴趣的:(Android大图片引起的内存问题(OOM))