Android中大图片加载

1.1.1 加载大图片原理

Android虚拟机默认为每个应用分配的堆内存是16M,当在界面显示图片时,需要的内存空间不是按图片的实际大小来计算的,而是按像素点的多少乘以每个像素点占用的空间大小来计算的。图片加载到内存中需要把每一个像素都加载到内存中. 所以对内存的要求非常高,一不小心就会造成OOM(OutOfMemoryError)错误。下面通过一组计算来演示OOM为什么会发生,以及解决方案。
假设:当前有一张图片,大小仅为1M,但是其规格(图片的像素)为3648*2736,那么完全加载此图片的像素数=3648*2736=9980928。在Android中像素的表现形式有三种模式:
三种像素如下:
l ARGB_4444 : 2bytes 每个像素占据2个字节
l ARGB_8888 : 4bytes 每个像素占据4个字节
l RGB_565 : 4bytes 每个像素占据4个字节
假设采用ARGB_4444模式作为像素的表现形式,则该1M大小的图片在Android中占用的总空间为:
图片占用空间=总像素数 *像素的单位
=9980928* 2bytes
=19961856bytes
=19M>16M
通过计算大家发现就算采用每个像素占用2个字节的形式该1M的图片也需要分配19M的空间才能100%地将所有像素表现出来,这肯定会导致OOM的发生。
解决方案:
通过代码可以对图片进行按比例(之所以按照比例缩放是为了避免图片的畸变,比如一张图片长度缩小一半,那么高度也缩小一半就保证了图片不会畸变)缩放,这也是必须要进行的工作。
假设:
图片的宽和高: 3648 * 2736
屏幕的宽和高: 320 * 480
计算缩放比:
宽度缩放比例: 3648 / 320 = 11
高度缩放比例: 2736 / 480 = 5
比较宽和高的缩放比例,哪一个大用哪一个进行缩放,因此我们采用11作为该图片长和高的缩放比例。
计算缩放后的图片的宽和高:
宽=3648 / 11 = 331
高=2736 / 11 = 248
缩放后图片的宽和高: 331* 248
计算缩放后的图片在Android中占有的空间:
总空间=331* 248*像素点大小=882088 * 2bytes=160K
最终计算结果是160K,远远小于堆内存的16M上限,从而解决了OOM的发生。

 /**
     * 点击界面Button后调用该方法,实现加载大图片的业务逻辑
     *
     * @param view
     */
    public void load(View view) {
        /**
         * 最普通的方式(不对图片进行等比例缩放时)加载图片的两行核心代码
         */
        Bitmap bitmap = BitmapFactory.decodeFile("/mnt/sdcard/dog.jpg");
        iv.setImageBitmap(bitmap);

        //创建一个可选项对象,该对象用于配置图片的处理参数
        Options opts = new Options();
        // 将该参数设置为true则加载器不加载图片, 而是把图片的out(宽和高)的字段信息取出来
        opts.inJustDecodeBounds = true;
        /**
         * 加载图片,该方法只是从图片文件中读取图片的宽和高信息,而没有真正的加载到内存中
         * 参数1:是图片的地址
         * 参数2:是加载图片时的配置信息
         */
        BitmapFactory.decodeFile("/mnt/sdcard/dog.jpg", opts);
        //获取到图片的宽和高信息
        int imageWidth = opts.outWidth;
        int imageHeight = opts.outHeight;

        // 获取到屏幕对象
        Display display = getWindowManager().getDefaultDisplay();
        // 获取到屏幕的真是宽和高
        int screenWidth = display.getWidth();
        int screenHeight = display.getHeight();
        // 计算缩放比例
        int widthScale = imageWidth / screenWidth;
        int heightScale = imageHeight / screenHeight;
        //计算出最大的比例
        int scale = widthScale > heightScale ? widthScale : heightScale;
        // 使用缩放比例进行缩放加载图片
        opts.inJustDecodeBounds = false; // 加载器就会返回图片了
        // 配置该参数加载图片时BitmapFactory就会自动缩放图片
        opts.inSampleSize = scale;
        //重新加载图片,这次才是 真正的压缩加载图片
        Bitmap bm = BitmapFactory.decodeFile("/mnt/sdcard/me.jpg", opts);
        // 显示在屏幕上
        iv.setImageBitmap(bm);
    }
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值