Android学习笔记:如何高效显示图片,避免内存溢出 和 ImageView无法显示大尺寸的图片

因为手机的内存资源是有限的,每个app可使用的内存是受限的。而现在采用高分辨率拍的照片往往很大。如果加载时不注意方法,很有可能会引起java.lang.OutofMemoryError: bitmap size exceeds VM budget. 异常而导致app奔溃退出。

另外ImageView支持的图片大小也是受限制的,比如整个App虽然只放一张图片,该图片大小也没超过整个app的内存上限。但该图片大小超过了ImageView的最大值,这也是有问题的。这时需要采取方法,在加载图片时缩小加载图片的大小。具体的策略看下面的介绍。

在真正创建和加载(需要实际耗费内存)一个图片对象时,我们可以先获取图像的大小信息。这里我们可以用到BitmapFactory.Options这个类。BitmapFactory.Options这个类,有一个字段叫做 inJustDecodeBounds 。SDK中对这个成员的说明是这样的:
“If set to true, the decoder will return null (no bitmap), but the out... fields will still be set, allowing the caller to query the bitmap without having to allocate the memory for its pixels.”

也就是说,如果我们把它设为true,那么BitmapFactory.decodeFile(String path, Options opt)并不会真的返回一个Bitmap给你,但它会把它的宽,高等基本信息取回来给你,这样就不会实际分配图片所需的内存。代码例子如下:

BitmapFactory.Options options = new BitmapFactory.Options();
options.inJustDecodeBounds = true;
BitmapFactory.decodeResource(getResources(), R.drawable.pic1, options);
String text = "imageHeight:" + options.outHeight + ",imageWidth:"
	+ options.outWidth + ",imageType:" + options.outMimeType;
Toast.makeText(this, text, Toast.LENGTH_LONG).show();

知道了图片的大小后,就可以考虑是将整个完整的图片加载到内存中,或是将缩小版加载到内存中。因为手机屏幕本身大小就那么大,没必要将原尺寸的图片加载进去。比如如果屏幕的大小为128x96,则将大小为1024x76的图片加载进去就没有意义。

为了告诉解码器将图片压缩后加载到内存中,需要设置BitmapFactory.Options对象的 inSampleSize 属性。比如,一张2048*1536的图片,如果设置inSampleSize的值为4. 缩小后图片的尺寸变为 512x384。占用的内存由原来的12M变为0.75M。下面的方法可以计算出一个图片应该设置的inSampleSize值:

public static int calculateInSampleSize(BitmapFactory.Options options,
			int reqWidth, int reqHeight) {
		// reqWidth、reqHeight是想要显示图片的大小,如屏幕的大小或ImageView控件的大小
		final int height = options.outHeight;
		final int width = options.outWidth;
		int inSampleSize = 1;

		if (height > reqHeight || width > reqWidth) {
			//说明图片的真实大小,大于需要显示的大小,则需要缩小图片
			final int halfHeight = height / 2;
			final int halfWidth = width / 2;
			while ((halfHeight / inSampleSize) > reqHeight
					&& (halfWidth / inSampleSize) > reqWidth) {
				inSampleSize *= 2;
			}
		}

		return inSampleSize;
	}

  有了适当的inSampleSize值后,就可以真正的执行加载图片的操作了。代码如下:

 final BitmapFactory.Options options = new BitmapFactory.Options();
 options.inSampleSize = 2;
 options.inJustDecodeBounds = false;
 Bitmap bitmap =  BitmapFactory.decodeResource(res, resId, options);

 ImageView imageView = (ImageView)findViewById(R.id.imageView1);
 imageView.setImageBitmap(bitmap);

实际上对于大的图片,通过使用inSampleSize将图片变小后加载到内存中,只要不是变了非常小,不会影响视觉效果。但来的好处是显而意见的,会大大降低对内存的使用。一般对于大尺寸的照片(如用手机自身拍的),将inSampleSize设置为4一般不会影响视觉效果。

  

时间: 2024-12-26 05:56:47

Android学习笔记:如何高效显示图片,避免内存溢出 和 ImageView无法显示大尺寸的图片的相关文章

Android学习笔记--使用HttpURLConnection实现网络下载效果,附带进度条显示

下面我就直接上代码了,因为代码中我已经写了非常详细的注释 1 package com.wuxianedu.httpdemo; 2 3 import android.app.ProgressDialog; 4 import android.content.Intent; 5 import android.net.Uri; 6 import android.os.AsyncTask; 7 import android.os.Handler; 8 import android.os.Message;

Android学习笔记(二二): 多页显示-Tag的使用

在手机屏幕中,Tab也是比较常用的,通常和List结合,例如我们手机的通信录.下面是Tag的结构. TabHost是整个Tab的容器,包括两部分,TabWidget和FrameLayout.TabWidget就是每个tab的标签,FrameLayout则是tab内容. 如果我们使用extends TabAcitivty,如同ListActivity,TabHost必须设置为@android:id/tabhost TabWidget必须设置android:id为@android:id/tabs F

Android学习笔记(七)——显示对话框窗口

显示对话框窗口 1.创建Dialog1项目,在activity_main.xml文件中添加一个Button: <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:l

Android学习笔记进阶之在图片上涂鸦(能清屏)

Android学习笔记进阶之在图片上涂鸦(能清屏) 2013-11-19 10:52 117人阅读 评论(0) 收藏 举报 HandWritingActivity.java [java] view plaincopy package xiaosi.handWriting; import android.app.Activity; import android.app.AlertDialog; import android.content.DialogInterface; import andro

Pro Android学习笔记(三三):Menu(4):Alternative菜单

什么是Alternative menu(替代菜单) 举个例子,Activity显示一个文本文件.如果用户想对文本文件进行编辑,Activity不提供编辑能力,但可由其他activity或者其他应用提供.我们将相关信息存储在一个intent中,例如该文本的Uri.这个intent可以匹配系统的多个应用,替代菜单将这些应用一一列出,菜单项的title就是该可被调用的activity的名字,图标也为该可被调用的activity的图表. 小例子说明 我们通过一个小例子进行学习,简单地打开一个URL:we

Android 学习笔记(二七):Menu

Menu由两种形式,Option menu和Context menu.前者是按下设备的Menu硬按钮弹出,后者是长按widget弹出. Option Menu 当我们按下Menu的硬件按钮时,Option Menu将被触发显示,最多可以显示6个选项的icon菜单,如果选项多于6个,第6个选项显示为“More“,点击可以进入扩展菜单.我们将在Android学习笔记(十一):Activity-ListView的例子一的基础上来学习Option Menu,也就是一个基于activity的菜单. 在这个

Android学习笔记:Home Screen Widgets(2):关于Widget

通过widget定义,我们在widget列表中看到了我们的TestWidget.当我们拖拽widget到主页时,假设在appwidet-provider中定义了android:configure的java类,在widget实例创建后会立即唤起配置activity. 这个activity主要完毕两个任务:1.配置初始化数据:2.将配置数据适配到widget实例中. 利用preference中存贮配置数据 widget数据能够保持在文件.Share preference,或者SQLite3中.wid

Android学习笔记二

17. 在ContentProvider中定义的getType()方法是定义URI的内容类型. 18. SQLiteDatabase类中的insert/delete/update/query方法其实也挺好用的,我在EquipmentProvider类中做了实现 19. Android专门有个单元测试项目(Android Test Project),在这个项目中,可以新建一个继承AndroidTestCase类的具体测试类来单元测试某个功能.我新建了一个AndroidTestProject项目,在

android学习笔记——使用QuickContactBadge关联联系人

本文大部分内容来自<疯狂android讲义>. QuickContactBadge继承了ImageView,因此它的本质也是图片,也可以通过android:src属性指定它显示的图片.QuickcontactBadge额外增加的功能是:该图片可以关联到手机中指定联系人,当用户单击该图片时,系统将会打开相应联系人的联系方式界面. 为了让QuickContactBadge与特定联系人关联,可以调用如下方法进行关联. assignContactFromEmail(String emailAddres