热门标签 | HotTags
当前位置:  开发笔记 > 运维 > 正文

浅析KJFrameForAndroid框架如何高效加载Bitmap

Bitmap是Android系统中的图像处理的最重要类之一。用它可以获取图像文件信息,进行图像剪切、旋转、缩放等操作,并可以指定格式保存图像文件。本文主要是从KJFrameForAndroid框架中分析高效加载Bitmap的方法

我们在写Android程序的时候,肯定会用到很多图片。那么对于图片的压缩处理自然是必不可少。为什么要压缩?我想这个问题不必在强调了,每个人在最初学习Android的时候肯定都会知道这么一个原因:我们编写的应用程序都是有一个最大内存限制,其中JAVA程序和C程序(NDK调用时)共享这一块内存大小,程序占用了过高的内存就容易出现OOM(OutOfMemory)异常。至于这个最大内存是多少,我们可以通过调用Runtime.getRuntime().maxMemory()方法验证一下。

正因为受到内存大小限制这一关键原因(其实不止这个原因,我想一张1M的图片和一张10k的图片,载入的速度必然也是不同的吧)。 如果你的控件大小只有40*40像素的大小,只是为了显示一张缩略图,这时候把一张1024*768像素的图片完全加载到内存中显然是不值得的,因此我们都会对图片做压缩处理。

BitmapFactory这个类提供了多个方法(decodeByteArray, decodeFile, decodeResource等)用于创建Bitmap对象,我们可以根据图片的来源选择合适的方法。然而这些方法会为已经读取的bitmap分配内存,这时如果是一张非常大的图片就会导致OOM出现。为此,每一种解析方法都提供了一个BitmapFactory.Options参数,可以通过将这个参数的inJustDecodeBounds属性设置为true就可以让解析方法禁止为bitmap分配内存,但是如此设置后BitmapFactory的返回值也不再是一个Bitmap对象,而是null。虽然Bitmap是null了,但是BitmapFactory.Options的outWidth、outHeight和outMimeType属性都会被赋值。使用这个技巧让我们可以在加载图片之前就获取到图片的长宽值和类型,从而根据情况对图片进行压缩。

BitmapFactory.Options optiOns= new BitmapFactory.Options(); 
 options.inJustDecodeBounds = true; 
 BitmapFactory.decodeFile(pathName, options);
 int h = options.outHeight; 
 int w = options.outWidth; 
 String type = options.outMimeType;

那么知道了图片的宽高,要如何压缩呢?BitmapFactory.Options有一个inSampleSize属性,这个int值表示图片的原宽高变为1/inSampleSize倍,如果原图是1024*768,inSampleSize=2,那么压缩后图片就变成了512*384。 最后将BitmapFactory.Options设置合适的inSampleSize值,并且记得将inJustDecodeBounds设置回false,再调用一次BitmapFactory相应的创建Bitmap的方法,并把Options传入,就可以得到压缩后的图片了。

这里有一个节选自开源Android应用开发框架KJFrameForAndroid中的一段代码

/**
 * 图片压缩处理(使用Options的方法)
 * 
 * @使用方法 首先你要将Options的inJustDecodeBounds属性设置为true,BitmapFactory.decode一次图片。
 *  然后将Options连同期望的宽度和高度一起传递到到本方法中。
 *  之后再使用本方法的返回值做参数调用BitmapFactory.decode创建图片。
 * 
 * @explain BitmapFactory创建bitmap会尝试为已经构建的bitmap分配内存
 * ,这时就会很容易导致OOM出现。为此每一种创建方法都提供了一个可选的Options参数
 * ,将这个参数的inJustDecodeBounds属性设置为true就可以让解析方法禁止为bitmap分配内存
 * ,返回值也不再是一个Bitmap对象, 而是null。虽然Bitmap是null了,但是Options的outWidth、
 * outHeight和outMimeType属性都会被赋值。
 * @param reqWidth
 *  目标宽度
 * @param reqHeight
 *  目标高度
 */
  public static BitmapFactory.Options calculateInSampleSize(
   final BitmapFactory.Options options, int reqWidth, int reqHeight) {
   // 源图片的高度和宽度
   final int height = options.outHeight;
   final int width = options.outWidth;
   int inSampleSize = 1;
   if (height > reqHeight || width > reqWidth) {
   // 计算出实际宽高和目标宽高的比率
   final int heightRatio = Math.round((float) height
    / (float) reqHeight);
   final int widthRatio = Math.round((float) width / (float) reqWidth);
   // 选择宽和高中最小的比率作为inSampleSize的值,这样可以保证最终图片的宽和高
   // 一定都会大于等于目标的宽和高。
   inSampleSize = heightRatio 

以上的方法适合使用在读取一个未知来源的图片时使用,因为你不知道这个未知来源图片的大小,那么还有一种方法是用在已经载入内存的图片,对已经载入内存的图片做压缩以后重新保存到本地,从而可以把一张原本1M大小的图片变成一张10K的图片。

这种方法的核心思想是首先将图片转成一个输出流,并记录输出流的byte数组大小,通过调用bitmap对象的compress方法,对图片做一次压缩以及格式化,并将byte数组大小与期望压缩的目标大小比对,得出压缩比率,并调用Bitmap的缩放方法,缩放计算出的压缩比率,从而得到压缩后的方法。

下面我们继续来看KJFrameForAndroid框架中的另一段代码:

/**
  * 图片压缩方法:(使用compress的方法)
  * 
  * @explain 如果bitmap本身的大小小于maxSize,则不作处理
  * @param bitmap
  *  要压缩的图片
  * @param maxSize
  *  压缩后的大小,单位kb
  */
 public static void imageZoom(Bitmap bitmap, double maxSize) {
  // 将bitmap放至数组中,意在获得bitmap的大小(与实际读取的原文件要大)
  ByteArrayOutputStream baos = new ByteArrayOutputStream();
  // 格式、质量、输出流
  bitmap.compress(Bitmap.CompressFormat.JPEG, 100, baos);
  byte[] b = baos.toByteArray();
  // 将字节换成KB
  double mid = b.length / 1024;
  // 获取bitmap大小 是允许最大大小的多少倍
  double i = mid / maxSize;
  // 判断bitmap占用空间是否大于允许最大空间 如果大于则压缩 小于则不压缩
  if (i > 1) {
   // 缩放图片 此处用到平方根 将宽带和高度压缩掉对应的平方根倍
   // (保持宽高不变,缩放后也达到了最大占用空间的大小)
   bitmap = scale(bitmap, bitmap.getWidth() / Math.sqrt(i),
     bitmap.getHeight() / Math.sqrt(i));
  }
 }
/***
  * 图片的缩放方法
  * 
  * @param src
  *  :源图片资源
  * @param newWidth
  *  :缩放后宽度
  * @param newHeight
  *  :缩放后高度
  */
 public static Bitmap scale(Bitmap src, double newWidth, double newHeight) {
  // 记录src的宽高
  float width = src.getWidth();
  float height = src.getHeight();
  // 创建一个matrix容器
  Matrix matrix = new Matrix();
  // 计算缩放比例
  float scaleWidth = ((float) newWidth) / width;
  float scaleHeight = ((float) newHeight) / height;
  // 开始缩放
  matrix.postScale(scaleWidth, scaleHeight);
  // 创建缩放后的图片
  return Bitmap.createBitmap(src, 0, 0, (int) width, (int) height,
    matrix, true);
 }

另外附上KJFrameForAndroid框架项目地址: https://github.com/kymjs/KJFrameForAndroid

或备用地址 http://git.oschina.net/kymjs/KJFrameForAndroid

有这方面需要的朋友可以下载下来自己研究下


推荐阅读
  • 在开发过程中,我最初也依赖于功能全面但操作繁琐的集成开发环境(IDE),如Borland Delphi 和 Microsoft Visual Studio。然而,随着对高效开发的追求,我逐渐转向了更加轻量级和灵活的工具组合。通过 CLIfe,我构建了一个高度定制化的开发环境,不仅提高了代码编写效率,还简化了项目管理流程。这一配置结合了多种强大的命令行工具和插件,使我在日常开发中能够更加得心应手。 ... [详细]
  • 提升 Kubernetes 集群管理效率的七大专业工具
    Kubernetes 在云原生环境中的应用日益广泛,然而集群管理的复杂性也随之增加。为了提高管理效率,本文推荐了七款专业工具,这些工具不仅能够简化日常操作,还能提升系统的稳定性和安全性。从自动化部署到监控和故障排查,这些工具覆盖了集群管理的各个方面,帮助管理员更好地应对挑战。 ... [详细]
  • 提升Android开发效率:Clean Code的最佳实践与应用
    在Android开发中,提高代码质量和开发效率是至关重要的。本文介绍了如何通过Clean Code的最佳实践来优化Android应用的开发流程。以SQLite数据库操作为例,详细探讨了如何编写高效、可维护的SQL查询语句,并将其结果封装为Java对象。通过遵循这些最佳实践,开发者可以显著提升代码的可读性和可维护性,从而加快开发速度并减少错误。 ... [详细]
  • 每日前端实战:148# 视频教程展示纯 CSS 实现按钮两侧滑入装饰元素的悬停效果
    通过点击页面右侧的“预览”按钮,您可以直接在当前页面查看效果,或点击链接进入全屏预览模式。该视频教程展示了如何使用纯 CSS 实现按钮两侧滑入装饰元素的悬停效果。视频内容具有互动性,观众可以实时调整代码并观察变化。访问以下链接体验完整效果:https://codepen.io/comehope/pen/yRyOZr。 ... [详细]
  • ButterKnife 是一款用于 Android 开发的注解库,主要用于简化视图和事件绑定。本文详细介绍了 ButterKnife 的基础用法,包括如何通过注解实现字段和方法的绑定,以及在实际项目中的应用示例。此外,文章还提到了截至 2016 年 4 月 29 日,ButterKnife 的最新版本为 8.0.1,为开发者提供了最新的功能和性能优化。 ... [详细]
  • 如何高效利用Hackbar插件提升网页调试效率
    通过合理利用Hackbar插件,可以显著提升网页调试的效率。本文介绍了如何获取并使用未包含收费功能的2.1.3版本,以确保在不升级到最新2.2.2版本的情况下,依然能够高效进行网页调试。此外,文章还提供了详细的使用技巧和常见问题解决方案,帮助开发者更好地掌握这一工具。 ... [详细]
  • Docker入门指南:初探容器化技术
    Docker入门指南:初探容器化技术摘要:Docker 是一个使用 Go 语言开发的开源容器平台,旨在实现应用程序的构建、分发和运行的标准化。通过将应用及其依赖打包成轻量级的容器,Docker 能够确保应用在任何环境中都能一致地运行,从而提高开发和部署的效率。本文将详细介绍 Docker 的基本概念、核心功能以及如何快速上手使用这一强大的容器化工具。 ... [详细]
  • 如何在任意浏览器中轻松安装并使用VSCode——Codeserver简易指南
    code-server 是一款强大的工具,允许用户在任何服务器上部署 VSCode,并通过浏览器进行访问和使用。这一解决方案不仅简化了开发环境的搭建过程,还提供了高度灵活的工作方式。用户只需访问 GitHub 上的官方仓库(GitHub-coder/code-server),即可获取详细的安装和配置指南,快速启动并运行 code-server。无论是个人开发者还是团队协作,code-server 都能提供高效、便捷的代码编辑体验。 ... [详细]
  • 观察 | 求职体验:收到录用通知的公司通常不深究技术细节,而那些详细追问的公司往往没有后续进展
    观察 | 求职体验:收到录用通知的公司通常不深究技术细节,而那些详细追问的公司往往没有后续进展 ... [详细]
  • 本文深入探讨了Spring框架中核心IOC容器的源代码实现,详细分析了其内部机制和工作原理。通过对关键类和方法的解读,揭示了IOC容器如何管理Bean的生命周期、依赖注入以及配置元数据的解析过程。此外,文章还讨论了容器启动时的初始化流程,帮助开发者更好地理解和使用Spring框架的核心功能。 ... [详细]
  • SSL 错误:目标主机名与备用证书主题名称不匹配
    在使用 `git clone` 命令时,常见的 SSL 错误表现为:无法访问指定的 HTTPS 地址(如 `https://ip_or_domain/xxxx.git`),原因是目标主机名与备用证书主题名称不匹配。这通常是因为服务器的 SSL 证书配置不正确或客户端的证书验证设置有问题。建议检查服务器的 SSL 证书配置,确保其包含正确的主机名,并确认客户端的证书信任库已更新。此外,可以通过临时禁用 SSL 验证来排查问题,但请注意这会降低安全性。 ... [详细]
  • 深入解析 Android 中 EditText 的 getLayoutParams 方法及其代码应用实例 ... [详细]
  • Presto:高效即席查询引擎的深度解析与应用
    本文深入解析了Presto这一高效的即席查询引擎,详细探讨了其架构设计及其优缺点。Presto通过内存到内存的数据处理方式,显著提升了查询性能,相比传统的MapReduce查询,不仅减少了数据传输的延迟,还提高了查询的准确性和效率。然而,Presto在大规模数据处理和容错机制方面仍存在一定的局限性。本文还介绍了Presto在实际应用中的多种场景,展示了其在大数据分析领域的强大潜力。 ... [详细]
  • 在日常的项目开发中,测试环境和生产环境通常采用HTTP协议访问服务。然而,从浏览器的角度来看,这种访问方式会被标记为不安全。为了提升安全性,当前大多数生产环境已经转向了HTTPS协议。本文将详细介绍如何在Spring Boot应用中配置SSL证书,以实现HTTPS安全访问。通过这一过程,不仅可以增强数据传输的安全性,还能提高用户对系统的信任度。 ... [详细]
  • 本文详细介绍了如何在Java Web服务器上部署音视频服务,并提供了完整的验证流程。以AnyChat为例,这是一款跨平台的音视频解决方案,广泛应用于需要实时音视频交互的项目中。通过具体的部署步骤和测试方法,确保了音视频服务的稳定性和可靠性。 ... [详细]
author-avatar
薇薇llfnc
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有