Android内存中的图片

图片在内存中的大小

Android.graphics.Bitmap类里有一个内部类Bitmap.Config类,在Bitmap类里createBitmap(intwidth, int height, Bitmap.Config config)方法里会用到,打开个这个类一看

枚举变量
public static final Bitmap.Config ALPHA_8
public static final Bitmap.Config ARGB_4444
public static final Bitmap.Config ARGB_8888
public static final Bitmap.Config RGB_565

说白了就ALPHA_8就是Alpha由8位组成
ARGB_4444就是由4个4位组成即16位,
ARGB_8888就是由4个8位组成即32位,
RGB_565就是R为5位,G为6位,B为5位共16位

由此可见:
ALPHA_8 代表8位Alpha位图
ARGB_4444 代表16位ARGB位图
ARGB_8888 代表32位ARGB位图
RGB_565 代表8位RGB位图

位图位数越高代表其可以存储的颜色信息越多,当然图像也就越逼真。

图片的分辨率是3776 * 2520,每一点又是由ARGB色组成,每个色素占4个Byte,所以这张图片加载到内存中需要消耗的内存为:
3776 * 2520 * 4byte = 38062080byte,会造成内存溢出,那么如何加载大图片呢?

如何加载大分辨率图片

有时候我们确实会需要加载一些大分辨率的图片,但是对于移动设备而言,哪怕加载能成功那么大的内存也是一种浪费(屏幕分辨率限制),所以就需要想办法把图片按照一定比率压缩,使分辨率降低,以至于又不需要耗费很大的堆内存空间,又可以最大的利用设备屏幕的分辨率来显示图片。这里就用到一个BitmapFactory.Options对象,下面来介绍它。
BitmapFactory.Options为BitmapFactory的一个内部类,它主要用于设定与存储BitmapFactory加载图片的一些信息。下面是Options中需要用到的属性:
inJustDecodeBounds:如果设置为true,将不把图片的像素数组加载到内存中,仅加载一些额外的数据到Options中。
outHeight:图片的高度。
outWidth:图片的宽度。
inSampleSize:如果设置,图片将依据此采样率进行加载,不能设置为小于1的数。例如设置为4,分辨率宽和高将为原来的1/4,这个时候整体所占内存将是原来的1/16。

示例代码:

 1 import android.os.Bundle;
 2 import android.os.Environment;
 3 import android.app.Activity;
 4 import android.graphics.Bitmap;
 5 import android.graphics.BitmapFactory;
 6 import android.graphics.BitmapFactory.Options;
 7 import android.view.Menu;
 8 import android.view.View;
 9 import android.view.WindowManager;
10 import android.widget.Button;
11 import android.widget.ImageView;
12 public class MainActivity extends Activity {
13     private Button btn_loadimage;
14     private ImageView iv_bigimage;
15      @Override
16     protected void onCreate(Bundle savedInstanceState) {
17         super.onCreate(savedInstanceState);
18         setContentView(R.layout.activity_main);
19          btn_loadimage = (Button) findViewById(R.id.btn_loadimage);
20         iv_bigimage = (ImageView) findViewById(R.id.iv_bigimage);
21          btn_loadimage.setOnClickListener(new View.OnClickListener() {
22              @Override
23             public void onClick(View v) {
24                 // Bitmap bitmap=BitmapFactory.decodeFile("/sdcard/a.jpg");
25                 // iv_bigimage.setImageBitmap(bitmap);
26                  BitmapFactory.Options opts = new Options();
27                 // 不读取像素数组到内存中,仅读取图片的信息
28                 opts.inJustDecodeBounds = true;
29                 BitmapFactory.decodeFile("/sdcard/a.jpg", opts);
30                 // 从Options中获取图片的分辨率
31                 int imageHeight = opts.outHeight;
32                 int imageWidth = opts.outWidth;
33                  // 获取Android屏幕的服务
34                 WindowManager wm = (WindowManager) getSystemService(WINDOW_SERVICE);
35                 // 获取屏幕的分辨率,getHeight()、getWidth已经被废弃掉了
36                 // 应该使用getSize(),但是这里为了向下兼容所以依然使用它们
37                 int windowHeight = wm.getDefaultDisplay().getHeight();
38                 int windowWidth = wm.getDefaultDisplay().getWidth();
39                  // 计算采样率
40                 int scaleX = imageWidth / windowWidth;
41                 int scaleY = imageHeight / windowHeight;
42                 int scale = 1;
43                 // 采样率依照最大的方向为准
44                 if (scaleX > scaleY && scaleY >= 1) {
45                     scale = scaleX;
46                 }
47                 if (scaleX < scaleY && scaleX >= 1) {
48                     scale = scaleY;
49                 }
50                  // false表示读取图片像素数组到内存中,依照设定的采样率
51                 opts.inJustDecodeBounds = false;
52                 // 采样率
53                 opts.inSampleSize = scale;
54                 Bitmap bitmap = BitmapFactory.decodeFile("/sdcard/a.jpg", opts);
55                 iv_bigimage.setImageBitmap(bitmap);
56              }
57         });
58     }
59 }
60   

加载缩略图

1. 使用inJustDecodeBounds,读bitmap的长和宽。
2. 根据bitmap的长款和目标缩略图的长和宽,计算出inSampleSize的大小。
3. 使用inSampleSize,载入一个大一点的缩略图A
4. 使用createScaseBitmap,将缩略图A,生成我们需要的缩略图B。
5. 回收缩略图A。

 需要注意的事情

createScaseBitmap如果原图和目标缩略图大小一致,那么不会生成一个新的bitmap直接返回bitmap,因此,回收的时候,要判断缩略图A是否就是缩略图B,如果说是的话,不要回收。

 1 /**
 2  * http://developer.android.com/training/displaying-bitmaps/load-bitmap.html
 3  */
 4 public class BitmapUtils {
 5     private static int calculateInSampleSize(BitmapFactory.Options options,
 6             int reqWidth, int reqHeight) {
 7         final int height = options.outHeight;
 8         final int width = options.outWidth;
 9         int inSampleSize = 1;
10         if (height > reqHeight || width > reqWidth) {
11             final int halfHeight = height / 2;
12             final int halfWidth = width / 2;
13             while ((halfHeight / inSampleSize) > reqHeight
14                     && (halfWidth / inSampleSize) > reqWidth) {
15                 inSampleSize *= 2;
16             }
17         }
18         return inSampleSize;
19     }
20
21     // 如果是放大图片,filter决定是否平滑,如果是缩小图片,filter无影响
22     private static Bitmap createScaleBitmap(Bitmap src, int dstWidth,
23             int dstHeight) {
24         Bitmap dst = Bitmap.createScaledBitmap(src, dstWidth, dstHeight, false);
25         if (src != dst) { // 如果没有缩放,那么不回收
26             src.recycle(); // 释放Bitmap的native像素数组
27         }
28         return dst;
29     }
30
31     // 从Resources中加载图片
32     public static Bitmap decodeSampledBitmapFromResource(Resources res,
33             int resId, int reqWidth, int reqHeight) {
34         final BitmapFactory.Options options = new BitmapFactory.Options();
35         options.inJustDecodeBounds = true;
36         BitmapFactory.decodeResource(res, resId, options); // 读取图片长款
37         options.inSampleSize = calculateInSampleSize(options, reqWidth,
38                 reqHeight); // 计算inSampleSize
39         options.inJustDecodeBounds = false;
40         Bitmap src = BitmapFactory.decodeResource(res, resId, options); // 载入一个稍大的缩略图
41         return createScaleBitmap(src, reqWidth, reqHeight); // 进一步得到目标大小的缩略图
42     }
43
44     // 从sd卡上加载图片
45     public static Bitmap decodeSampledBitmapFromFd(String pathName,
46             int reqWidth, int reqHeight) {
47         final BitmapFactory.Options options = new BitmapFactory.Options();
48         options.inJustDecodeBounds = true;
49         BitmapFactory.decodeFile(pathName, options);
50         options.inSampleSize = calculateInSampleSize(options, reqWidth, reqHeight);
51         options.inJustDecodeBounds = false;
52         Bitmap src = BitmapFactory.decodeFile(pathName, options);
53         return createScaleBitmap(src, reqWidth, reqHeight);
54     }
55 }

时间: 2024-10-19 10:59:02

Android内存中的图片的相关文章

Android工程中加入图片,报错cannot be resolved or is not a field

SDK和ADT为22.6.2版本 工程为4.4.2 今天在写Android代码的时候,往工程中加入了几张图片,然后在代码中使用R.drawable调用时,一直报错 cannot be resolved or is not a field 然后我查看了gen目录下的R.java文件,发现里面已经有我加入的图片资源ID了,觉得很奇怪,一般是无法生成R.java文件的时候才会出现这种现象啊 在网上查了资料也未见有可以解决我这个问题的方法,然后我就把我的代码从头到尾重新看了一遍,开始也没有发现什么异常,

Android:WebView中对图片注册上下文菜单

前言 今天一朋友问我一个问题,就是如何在WebView控件中的图片增加上下文菜单,以便增加保存图片等功能.今天就给他简单做了一个演示Demo,现写下来,给有相同问题的朋友提供些许思路吧. 概要实现 其实这个功能很简单,没有太复杂的东西,就是对WebView的控件的使用,一是给WebView注册了上下文菜单事件,二是在响应事件中去判断事件源的类型,如果是图片类型,则把url取出来 注册上下文菜单事件 这个就比较简单了通过下面的代码即可完成. WebView vw = (WebView) findV

win7(旗舰版)下,OleLoadPicture 加载内存中的图片(MagickGetImageBlob),返回值 &lt; 0

昨天去三哥家,想把拍好的照片缩小一下,我用很久前写的一个软件进行缩小,然后进行一次效果预览,这个时候弹出: Call OleLoadPicture Fail - loadPictureFromMW 奇怪,这个图片批量处理软件我一直在用,在 xp系统下从来没出现过这样的提示,为什么在 win7下无法正常进行图片的效果预览? 更让我接受不了的是,从来没有用户或朋友向我反应这个问题,难道他们都不用 win7? 今天下午问了一个朋友借了台win7的笔记本,通过一下午的测试,解决了这个问题. 之前的代码思

Android EditText中插入图片并响应点击事件

EditText中插入图片基本就是两种方法: 1,通过Html.fromHtml(..)来实现 [mw_shl_code=java,true]eText.append(Html.fromHtml("<img src='" + R.drawable.ohoh + "'/>", imageGetter, null));[/mw_shl_code] 2,通过ImageSpan与SpannableString/SpannableStringBuilder. [m

Android开发中根据图片名称获取在drawable中的ID

在Android开发中图片资源是必不可少的,如ImageView需要图片资源的ID,ImageButton需要资源的ID等等,我们可以用R.drawable.XXX可以获取图片资源的ID,但是,在某些时候,这样做很费时,我们想动态的获得资源ID,比如说,我传入一个图片名称的字符串,根据字符串来获得资源的ID这样就很方便了,没错,这样确实很方便,我们如果对图片的名称稍加改动,比如用img1.png,img2.png,img3.png...这样就可以在一个循环之内获得所有的ID,对开发来说少写的就不

知道内存中一个图片的指针IntPtr大小,转换成图片显示

//nSize 为总长度//pImageData 为总数据//nImageSize //一个图片的长度 byte[] _bytes = new byte[nImageSize];// //IntPtr infosIntptr = Marshal.AllocHGlobal(pImageData); Marshal.Copy(pImageData, _bytes, 0, nImageSize);//复制 PtrMoveSize(ref pImageData, nImageSize);//丛总内存中去

android内存优化之图片优化

对图片本身进行操作.尽量不要使用setImageBitmap.setImageResource.BitmapFactory.decodeResource来设置一张大图,因为这些方法在完成decode后,最终都是通过java层的createBitmap来完成的,需要消耗更多内存.因此,改用先通过BitmapFactory.decodeStream方法,创建出一个bitmap,再将其设为ImageView的source,decodeStream最大的秘密在于其直接调用JNI>>nativeDeco

android内存优化之图片压缩和缓存

由于手机内存的限制和网络流量的费用现在,我们在加载图片的时候,必须要做好图片的压缩和缓存. 图片缓存机制一般有2种,软引用和内存缓存技术. 1.压缩图片:压缩图片要既不能模糊,也不能拉伸图片. 图片操作的时候,最常用的是BitmapFactory,现在看看如何压缩图片. Bitmapfactory.Options options= new BitmapFactory.Options(); options.inJustDecodeBounds = true;// 如果设置这个参数为ture,就不会

内存中一个图片的指针IntPtr

你知道图像的指针,如果知道图像大小,就可以用 byte[] data = new byte[picLength];Marshal.Copy(img, data, 0, data.length);using(MemoryStream ms = new MemoryStream(data)){  imgbox = Image.FromSteam(ms);} http://bbs.csdn.net/topics/350242149