Android 内存管理研究
1. 内存管理基础知识
http://www.cnblogs.com/xingfuzzhd/p/3485924.html
- 1. mImageView.setImageResource(R.drawable.my_image);这段代码会调用 BitmapFactory.decodeStream() 生成一个 Bitmap。所以不要以为它比自己创建 Bitmap 节省内存。
3. 实际测试:
我使用了多种调用图片的方法来测试:
第一种:
// 直接载入资源 id
ImageView image = (ImageView) findViewById(R.id.imageView1)
第二种:
创建一个函数,根据 id 载入图片。杀死 Activity 时回收 Bitmap
public BitmapDrawable getImageBitmap(int idImage) {
  Bitmap bitmap = BitmapFactory.decodeResource(context.getResources(), idImage);
  images.add(bitmap);
  return new BitmapDrawable(context.getResources(), bitmap);
}
第三种:
创建一个类
/**
* solution for OOM ------> compress bitmap
*/
public class BitmapUtils {
private static final String TAG = "BitmapUtils";
/**
* calulate the compress rate
*/
private static int calculateInSampleSize(BitmapFactory.Options opts,int reqHeight,int reqWidth) {
if(opts == null) {
return -1;
}
int width = opts.outWidth;
int height = opts.outHeight; int sampleSize = 1; if(width > reqWidth || height > reqHeight) {
int heightRatio = (int) (height/(float)reqHeight);
int widthRatio = (int) (width/(float)reqWidth);
sampleSize = (heightRatio > widthRatio) ? widthRatio : heightRatio;
}
return sampleSize;
} /**
* compress an image refer to the goal dimension
* @param res
* @param resId
* @param reqWidth
* @param reqHeight
* @return Bitmap
*/
public static Bitmap decodeSampledBitmapFromResource(Resources res,int resId,int reqWidth,int reqHeight) {
BitmapFactory.Options opts = new BitmapFactory.Options();
opts.inJustDecodeBounds = true;
BitmapFactory.decodeResource(res, resId, opts);
int sampleSize = calculateInSampleSize(opts, reqHeight, reqWidth);
Log.i(TAG,"before[width:"+opts.outWidth+",height:"+opts.outHeight+"]");
opts.inJustDecodeBounds = false;
opts.inSampleSize = sampleSize;
/* newly added */
opts.inPreferredConfig = Bitmap.Config.RGB_565;
opts.inPurgeable = true;
opts.inInputShareable = true;
/* newly added */
Log.i(TAG,"insamplesize="+sampleSize);
Bitmap bitmap = BitmapFactory.decodeResource(res, resId, opts);
Log.i(TAG,"after[width:"+bitmap.getWidth()+",height:"+bitmap.getHeight()+"]");
return bitmap;
} /**
* compress an image refer to the goal dimension
*
* @param data image byte array
* @param reqWidth
* @param reqHeight
* @return
*/
public static Bitmap decodeSampledBitmapFromByteArray(byte[] data,int reqWidth,int reqHeight) {
BitmapFactory.Options opts = new BitmapFactory.Options();
opts.inJustDecodeBounds = true;
BitmapFactory.decodeByteArray(data, 0, data.length, opts);
Log.i(TAG,"before[width:"+opts.outWidth+",height:"+opts.outHeight+"]");
opts.inSampleSize = calculateInSampleSize(opts, reqHeight, reqWidth);
Log.i(TAG,"insamplesize="+opts.inSampleSize);
opts.inJustDecodeBounds = false;
Bitmap bitmap = BitmapFactory.decodeByteArray(data, 0, data.length, opts);
Log.i(TAG,"after[width:"+bitmap.getWidth()+",height:"+bitmap.getHeight()+"]");
return bitmap;
} public static Bitmap decodeResourceWithLowMemory(Context context, int resId) {
BitmapFactory.Options opt = new BitmapFactory.Options();
// Bitmap.Config.ALPHA_8,Bitmap.Config.ARGB_4444,Bitmap.Config.RGB_565
// 设置这几个参数效果都差不多,确实相当省内存啊,而且还跟原图清晰度相当
opt.inPreferredConfig = Bitmap.Config.RGB_565;
opt.inPurgeable = true;
opt.inInputShareable = true;
return BitmapFactory.decodeResource(context.getResources(), resId, opt);
} public static Bitmap loadToImageViewFromResource(Resources res,int resId,ImageView imageView) {
BitmapFactory.Options opts = new BitmapFactory.Options();
opts.inJustDecodeBounds = true;
BitmapFactory.decodeResource(res, resId, opts);
int reqHeight = imageView.getLayoutParams().height;
int reqWidth = imageView.getLayoutParams().width;
int sampleSize = calculateInSampleSize(opts, reqHeight, reqWidth);
Log.i(TAG,"before[width:"+opts.outWidth+",height:"+opts.outHeight+"]");
opts.inJustDecodeBounds = false;
opts.inSampleSize = sampleSize;
/* newly added */
opts.inPreferredConfig = Bitmap.Config.RGB_565;
opts.inPurgeable = true;
opts.inInputShareable = true;
/* newly added */
Log.i(TAG,"insamplesize="+sampleSize);
Bitmap bitmap = BitmapFactory.decodeResource(res, resId, opts);
Log.i(TAG,"after[width:"+bitmap.getWidth()+",height:"+bitmap.getHeight()+"]");
return bitmap;
} }
4. 测试过程:
创建两个 Activity,每个都载入几个大图片,然后两个 Activity 之间不停的切换。
第一种方法,切换多次后 OOM (out of memory)。 第二种方法,更快的 OOM。
第三种,如果压缩率够高的话,几乎不会 OOM。比如原始图片为 2048 * 1600,然后目标 ImageView 的大小为 200* 150。最后内存消耗将大大减小。然后如果
两者分辨率差距不大,最后内存消耗将不会被优化,最后也将 OOM。
5. 原理
通常,我们使用BitmapFactory.decodeResource()方法来从资源文件中读取一张 图片并生成一个 Bitmap。但如果使用一个BitmapFactory.Options对象,并把该对象的inJustDecodeBounds属性设 置为true,decodeResource()方法就不会生成 Bitmap 对象,而仅仅是读取该图片的尺寸和类型信息。所以在前面的类的一些方法中,我们用 BitmapFactory decode了两次图片,但第一次只是获取信息而没有生成 Bitmap 对象,所以没有未回收的 Bitmap 出现。
6. 一些推荐的解决方案
弱引用,LruCache,或者开源项目 :Android-Universal-Image-Loader
延伸:一个Bitmap到底占用多大内存?系统给每个应用程序分配多大内存?
· Bitmap占用的内存为:像素总数 * 每个像素占用的内存。在Android中,Bitmap有四种像素类型:ARGB_8888、ARGB_4444、ARGB_565、ALPHA_8,他 们每个像素占用的字节数分别为4、2、2、1。因此,一个2000*1000的ARGB_8888类型的Bitmap占用的内存为 2000*1000*4=8000000B=8MB。
· Android根据设备屏幕尺寸和dpi的不同,给系统分配的单应用程序内存大小也不同,具体如下表(表格取自Android 4.4 Compatibility Definition Document (CDD)):
| 屏幕尺寸 | DPI | 应用内存 | 
| small / normal / large | ldpi / mdpi | 16MB | 
| small / normal / large | tvdpi / hdpi | 32MB | 
| small / normal / large | xhdpi | 64MB | 
| small / normal / large | 400dpi | 96MB | 
| small / normal / large | xxhdpi | 128MB | 
| xlarge | mdpi | 32MB | 
| xlarge | tvdpi / hdpi | 64MB | 
| xlarge | xhdpi | 128MB | 
| xlarge | 400dpi | 192MB | 
| xlarge | xxhdpi | 256MB | 
Android 内存管理研究的更多相关文章
- Android内存管理机制之一:low memory killer
		转载自http://www.miui.com/thread-29268-1-1.html 准备写这个专题之前,心里是有点忐忑的.首先Android内存管理机制相当复杂,想要讲清楚比较困难:其次对于绝大 ... 
- Android 内存管理分析(四)
		尊重原创作者,转载请注明出处: http://blog.csdn.net/gemmem/article/details/8920039 最近在网上看了不少Android内存管理方面的博文,但是文章大多 ... 
- 移动端测试===Android内存管理: 理解App的PSS
		Android内存管理: 理解App的PSS 原文链接:http://www.littleeye.co/blog/2013/06/11/android-memory-management-unders ... 
- 浅谈Android内存管理
		最近在网上看了不少Android内存管理方面的博文,但是文章大多都是就单个方面去介绍内存管理,没有能全局把握,缺乏系统性阐述,而且有些观点有误,仅仅知道这些,还是无法从整体上理解内存管理,对培养系统优 ... 
- [Android Memory]   Android内存管理、监测剖析
		转载自:http://blog.csdn.net/anlegor/article/details/23398785 Android内存管理机制: Android内存管理主要有:LowMemory Ki ... 
- Android——内存管理基础
		内存收集概念 内存垃圾收集器(garbage collector) 概念:自定内存管理. 功能:分配内存.保证所有被引用的对象还在内存中.可以释放在运行的代码中不再引用的对象的内存. 垃圾收集器避免了 ... 
- Android内存管理
		首先Android理机制相当复杂.想要讲清楚比較困难.其次对于绝大多数用户来说.仅仅关心内存够不够用,至于内存怎样管理的这样的技术细节,不是用户须要去考虑的,写这样一个专题有没有意义?毕竟我们是用手机 ... 
- Android 内存管理 &Memory Leak & OOM 分析
		转载博客:http://blog.csdn.net/vshuang/article/details/39647167 1.Android 进程管理&内存 Android主要应用在嵌入式设备当中 ... 
- 【原创】Android内存管理-OnTrimMemory
		Application中有两个与内存管理相关的方法:onLowMemory()和 onTrimMemory(int level),源码如下 @CallSuper public void onLowMe ... 
随机推荐
- MVC框架介绍
			第一,建立一个解决方案然后在该解决方案下面新建mvc空项目. 第二,下面先对该项目的一些文件进行介绍: MVC项目文件夹说明: 1.(App_Data):用来保存数据文件,比如XML文件等 2.(Ap ... 
- selenium webdriver——JS滚动到指定位置
			1.DOM滚动方法 1.scrollIntoView(alignWithTop) 滚动浏览器窗口或容器元素,以便在当前视窗的可见范围看见当前元素.如果alignWithTop为true,或者省略它, ... 
- 「小程序JAVA实战」小程序的springboot后台拦截器(61)
			转自:https://idig8.com/2018/09/24/xiaochengxujavashizhanxiaochengxudespringboothoutailanjieqi60/ 之前咱们把 ... 
- UNITY 复制对象后局部坐标和世界坐标的变化问题
			void Start () { var pgo = transform.Find ("Button").gameObject; obtn = Instantiate (pgo); ... 
- centos 安装hue 4.0
			Hue是Cloudera开源的一个Hadoop UI,由Cloudera Desktop演化而来.面向用户提供方便的UI用于平时的Hadoop操作中.Apache Ambari面向的是管理员,用于安装 ... 
- Python dict() 函数
			Python dict() 函数 Python 内置函数 描述 dict() 函数用于创建一个字典. 语法 dict 语法: class dict(**kwarg) class dict(mappi ... 
- sqlserver 必须声明标量变量 "***"。
			发现在navicat premium上执行报这个异常,在sqlserver上不报,想到我之前的文章用存储过程时mysql里有个分割符,去掉“:”果然执行成功. DECLARE @countlmc IN ... 
- 判断单选框选中不成功,$("#xx").attr("checked")undefined
			判断单选框选中状态,各种都不行,受到https://www.cnblogs.com/yxwkf/p/4853014.html 的启发,相关引用: 原来.在jquery1.6版本号便对此做出了改动: [ ... 
- 手工kill掉VNC进程的故障处理
			1.模拟Kill掉已经启动的VNC服务 1)启动桌面1的服务 [root@testdb ~]# vncserver :1 New 'testdb:1 (root)' desktop is testdb ... 
- [Jmeter]如何才能通过ant运行jmeter
			在开始运行build.xml之前,还有一步必须要做,那就是将JMeter所在目录下extras子目录里的ant-JMeter-1.1.1.jar复制到Ant所在目录lib子目录之下,这样Ant运行时才 ... 
