Android--加载大分辨率图片到内存

简介:

还原堆内存溢出的错误

  首先来还原一下堆内存溢出的错误。首先在SD卡上放一张照片,分辨率为(3776 X 2520),大小为3.88MB,是我自己用相机拍的一张照片。应用的布局很简单,一个Button一个ImageView,然后按照常规的方式,使用BitmapFactory加载一张照片并使用一个ImageView展示。

  代码如下:

复制代码
1 btn_loadimage.setOnClickListener(new View.OnClickListener() {
2             
3             @Override
4             public void onClick(View v) {
5                 Bitmap bitmap=BitmapFactory.decodeFile("/sdcard/a.jpg");
6                 iv_bigimage.setImageBitmap(bitmap);
7             }
8 }
复制代码

  当点击按钮后,程序会报错,查看日志为:

  先来分析一下这个错误,首先dalvikvm(Android虚拟机)发现需要的内存38MB大于应用的堆内存24MB,这个时候尝试使用软加载的方式加载数据,我们知道当内存不足的时候dalvikvm会自动进行GC(Garbage Collection),大概清理了55k的空间出来,耗时203毫秒,但是内存还是不够,所以最后发生堆内存溢出的错误。

 

分析堆内存溢出

  Android系统主要用于低能耗的移动设备,所以对内存的管理有很多限制,一个应用程序,Android系统缺省会为其分配最大16MB(某些机型是24MB)的空间作为堆内存空间,我这里使用的模拟器调试的,这个模拟器被设定为24MB,可以在Android Virtual Device Manager中查看到。

  而这里的图片明明只有3.88MB,远远小于Android为应用分配的堆内存,而加载到内存中,为什么需要消耗大约38MB的内存呢?

  我们都知道,图片是由一个一个点分布组成的(分辨率),通常加载这类数据都会在内存中创建一个二维数组,数组中的每一项代表一个点,而这个图片的分辨率是3776 * 2520,每一点又是由ARGB色组成,每个色素占4个Byte,所以这张图片加载到内存中需要消耗的内存为:

  3776 * 2520 * 4byte = 38062080byte

  大约需要38MB的内存才能正确加载这张图片,这就是上面错误描述需要38MB的内存空间,大小略有出入,因为图片还有一些Exif信息需要存储,会比仅靠分辨率计算要大一些。

 

如何加载大分辨率图片

  有时候我们确实会需要加载一些大分辨率的图片,但是对于移动设备而言,哪怕加载能成功那么大的内存也是一种浪费(屏幕分辨率限制),所以就需要想办法把图片按照一定比率压缩,使分辨率降低,以至于又不需要耗费很大的堆内存空间,又可以最大的利用设备屏幕的分辨率来显示图片。这里就用到一个BitmapFactory.Options对象,下面来介绍它。

  BitmapFactory.Options为BitmapFactory的一个内部类,它主要用于设定与存储BitmapFactory加载图片的一些信息。下面是Options中需要用到的属性:

  • inJustDecodeBounds:如果设置为true,将不把图片的像素数组加载到内存中,仅加载一些额外的数据到Options中。
  • outHeight:图片的高度。
  • outWidth:图片的宽度。
  • inSampleSize:如果设置,图片将依据此采样率进行加载,不能设置为小于1的数。例如设置为4,分辨率宽和高将为原来的1/4,这个时候整体所占内存将是原来的1/16。

  

示例Demo

  下面通过一个简单的Demo来演示上面提到的内容,代码中注释比较清晰,这里就不再累述了。

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

  效果展示:

 

  源码下载

 

总结

  本篇博客到这里就讲解了如何加载一个大分辨率的图片到内存中并使用它。不过一般好一点的图片处理软件,都会有图片放大功能,如果仅做此处理,单纯的把处理后的图片放大,会影响显示效果,图片还原度不高。一般会重新获取放大区域的图片的分辨率像素数组,然后重新处理加载到内存中进行显示。



本文转自承香墨影博客园博客,原文链接:http://www.cnblogs.com/plokmju/p/android_LoadBigImage.html,如需转载请自行联系原作者


相关文章
|
21天前
|
存储 开发工具 Android开发
构建高效的Android应用:从内存管理到用户界面
【5月更文挑战第29天】 随着智能手机的普及,Android应用的开发变得日益重要。然而,许多开发者在开发过程中忽视了性能优化,导致应用运行缓慢,用户体验差。本文将深入探讨如何通过有效的内存管理和用户界面优化,提升Android应用的性能。我们将详细介绍内存泄漏的原因和解决方案,以及如何使用Android的新特性来创建流畅的用户界面。无论你是新手还是经验丰富的开发者,都可以从本文中获得有用的技巧和建议。
|
12天前
|
XML API 开发工具
Android Bitmap 加载与像素操作
Android Bitmap 加载与像素操作
11 2
|
13天前
|
编解码 缓存 Android开发
构建高效的Android应用:从内存优化到响应式设计
【5月更文挑战第37天】 在竞争激烈的移动应用市场中,一个高效、流畅的Android应用是吸引和保留用户的关键。本文将深入探讨构建高效Android应用的多个关键方面,包括内存优化策略、布局性能和响应式设计原则。我们将通过具体的技术实践和案例分析,揭示如何提升应用性能,减少资源消耗,并确保在不同设备上的兼容性和用户体验一致性。
|
15天前
|
消息中间件 存储
【消息队列开发】 实现内存加载
【消息队列开发】 实现内存加载
|
19天前
|
缓存 监控 Android开发
Android 开发中的内存优化策略
【5月更文挑战第30天】在移动应用的开发过程中,性能和用户体验始终是核心关注点。对于基于Android平台的应用程序,有效的内存管理是确保流畅运行和优异性能的关键因素之一。本文将深入探讨Android开发中常见的内存问题,并提出一系列实用的内存优化策略。我们将从内存泄漏的识别与防止开始,到合理使用内存缓存技巧,以及高效的数据结构选择等方面进行详细阐述。通过这些策略的实施,开发者可以显著减少应用的内存占用,提升应用的稳定性和响应速度,进而改善最终用户的体验。
|
21天前
|
移动开发 安全 Android开发
构建高效Android应用:采用Kotlin进行内存优化
【5月更文挑战第29天】 在移动开发领域,性能优化一直是开发者关注的焦点。特别是对于Android应用而言,内存管理是影响应用性能和用户体验的关键因素之一。近年来,Kotlin作为官方推荐的开发语言,以其简洁的语法和强大的功能受到广大开发者的青睐。本文将深入探讨如何通过Kotlin语言的特性来优化Android应用的内存使用,从而提升应用的性能表现。我们将从内存泄露检测、对象创建与销毁策略,以及数据结构的合理选择等方面入手,为读者提供一系列实用的优化建议。
|
22天前
|
移动开发 监控 Android开发
构建高效Android应用:从内存优化到电池寿命代码之美:从功能实现到艺术创作
【5月更文挑战第28天】 在移动开发领域,特别是针对Android系统,性能优化始终是关键议题之一。本文深入探讨了如何通过细致的内存管理和电池使用策略,提升Android应用的运行效率和用户体验。文章不仅涵盖了现代Android设备上常见的内存泄漏问题,还提出了有效的解决方案,包括代码级优化和使用工具进行诊断。同时,文中也详细阐述了如何通过减少不必要的后台服务、合理管理设备唤醒锁以及优化网络调用等手段延长应用的电池续航时间。这些方法和技术旨在帮助开发者构建更加健壮、高效的Android应用程序。
|
22天前
|
监控 Android开发 UED
构建高效的Android应用:从内存管理到用户体验
【5月更文挑战第28天】 在移动开发的世界中,打造一个既快速又高效的Android应用是每个开发者追求的目标。本文将深入探讨如何通过合理的内存管理策略、优化的代码实践和用户界面设计的提升来增强应用性能。我们将剖析内存泄漏的根源,提供解决方案,探索Kotlin与Java在性能上的差异,并分析如何利用Android系统提供的工具监控和改善应用表现。此外,我们还将讨论Material Design的应用以及其对用户体验的影响。通过这些技术的综合运用,你将能够为用户提供更流畅、响应更快的使用体验。
|
22天前
|
监控 安全 网络安全
网络安全与信息安全:防护之道与加密技术构建高效Android应用:从基础到高级的内存优化策略
【5月更文挑战第27天】在数字化时代,数据成为了新的货币。然而,随着信息技术的蓬勃发展,网络安全漏洞和信息泄露事件层出不穷,对个人隐私和企业安全构成了严重威胁。本文将深入探讨网络安全的重要性,分析当前常见的网络攻击方式,并重点分享关于加密技术和提升安全意识的知识。通过阅读本文,读者将获得如何有效防御网络威胁、保护个人和企业信息安全的策略。
|
23天前
|
存储 缓存 算法
深入理解操作系统内存管理:分页系统的优势与挑战构建高效Android应用:探究Kotlin协程的优势与实践
【5月更文挑战第27天】 在现代计算机系统中,内存管理是操作系统的核心功能之一。分页系统作为一种内存管理技术,通过将物理内存划分为固定大小的单元——页面,为每个运行的程序提供独立的虚拟地址空间。这种机制不仅提高了内存的使用效率,还为多任务环境提供了必要的隔离性。然而,分页系统的实现也带来了一系列的挑战,包括页面置换算法的选择、内存抖动问题以及TLB(Translation Lookaside Buffer)的管理等。本文旨在探讨分页系统的原理、优势及其面临的挑战,并通过分析现有解决方案,提出可能的改进措施。