在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虚拟机的堆内存分配
五:自定义堆内存大小
而对于前面我们找到的问题,我们对第一个和第二个解决办法进行整合就能够解决这个问题,(写代码):
在项目中主要将前两种方法进行了整合。
在通过包名获取bitmap时,首先在hardManager中判断是否包含该包名,如果有,直接取出;如果没有,则在softManager中判断,有取出;没有就通过下面方法获取。获取之后在插入到hardManager中。在插入时,做判断只保证hardManager的容量为10,如果大于10,则将最早插入的删除。