Android Bitmap转换WebP图片导致损坏的分析及解决方案
背景
作为移动领域所力推的图片格式,WebP
图片在商业领域证明了其应有的价值。基于其他格式的横向对比,其在压缩性能表现,及还原度极为优秀,节省大量的带宽开销。基于可观的效益比,团队早前已开始磋商将当前图片资源迁移至.webp资源。
然而对于Android而言,加载.webp
图片所消耗的时间比.jpg
及.png
要慢数倍。对于这点而言是无法忍受的。因此解决方案是:
从网络拿到.webp
数据流 -> Bitmap
通过.png
格式保存到本地
注意,整个过程必须在子线程执行。这样,在使用了WebP
节省了带宽的同时,下一次加载图片的速度也不会受到影响。
但在客户端实现的最后阶段,出现了一些问题。
问题重现
对于上述的解决方案,隐去业务复杂性,我用以下示例来展示:
private void saveImage(String uri, String savePath) throws IOException {
// 创建连接
HttpURLConnection conn = createConnection(uri);
// 拿到输入流,此流即是图片资源本身
InputStream imputStream = conn.getInputStream();
// 指使Bitmap通过流获取数据
Bitmap bitmap = BitmapFactory.decodeStream(imputStream);
File file = new File(savePath);
OutputStream out = new BufferedOutputStream(new FileOutputStream(file.getCanonicalPath()), BUFFER_SIZE);
// 指使Bitmap以相应的格式,将当前Bitmap中的图片数据保存到文件
if (bitmap.compress(Bitmap.CompressFormat.PNG, 100, out)) {
out.flush();
out.close();
}
}
上述代码意图明显:拿到流,将该流通过decodeStream(InputStream)
方法传送到Bitmap
,随后以.png
格式存储到本地。
在很长一段时间内,该代码运作良好。直到有一天,在某国产机型上做测试的时候,发现图片保存到本地后出现了损坏。
那些保存到本地出现损坏的图片,长这样:
在这张样图中,图片的下半部分出现了缺失。在随后的循环测试中,每张图片的缺失程度大小不一,从完整到全黑都有。
分析
对于这种情况,第一猜想可能是网络返回的数据流有问题。但在随后的排查中,发现InputStream
数据流是完整的。随后开始对图片本身进行分析。
对文件差异进行分析是一种好办法。在这里,使用Beyond Compare以不同的方式进行分析。于是准备了两张图片,一张成功从.webp
转为.png
,另一张也从.webp
转为.png
,但是出现缺失黑块。
现在,通过Picture Compare模式直观地对比两张图片:
在这里,左侧为完整图片,右侧为存在数据缺失的图片,下方为差异标记:红色区域为两张图片的差异之处。
可以观察到,相对于完整图片而言,存在数据缺失的图片并非零散地缺失数据,而是从某一刻开始,数据便不复存在了。
为了进一步考究导致差异的根本原因,可以通过Hex Compare模式进行对比。也就是说,以十六进制的方式对比文件。现在,通过Hex Compare模式进行文件对比:
左侧的红条表示两个文件中二进制数据不一致的地方。
其中,左侧为完整的.png
文件,右侧为存在缺失黑块的.png
文件。观察缺失文件的十六进制数据,存在着大量的空值块(0x00000000),并且数据长度是短于完整文件的。同时,此现象与早前出现黑块的规律相似:大块的数据丢失,并非零散的缺失。
但是,文件的分析尚未结束。有一个非常重要的问题不要忽略了:
我们是打开了一张数据损坏的图像吗?
我们知道,如果一个图像文件的关键数据块出现损坏,该图像是无法被打开的。也就是说,如果一个图像文件能够被打开,说明该图像文件结构完整。
那么,如何分析一张图像的数据块是否完整?在这里,我们关心的是:那张缺失的图像,文件末尾写入成功了吗?
在这里有必要解释一下PNG
文件末尾的数据块是个什么东西。引用PNG
格式标准的官方说法(PNG格式块简述:w3.org):
Chunks can appear in any order, subject to the restrictions placed on each chunk type. (One notable restriction is that IHDR must appear first and IEND must appear last; thus the IEND chunk serves as an end-of-file marker.) Multiple chunks of the same type can appear, but only if specifically permitted for that type.
解释:在整个PNG文件中,用以标记文件开始的IHDR标记必须在文件的最开始,标记文件结束的IEND标记必须在文件的最末端。对于其他数据块则没有顺序要求。
也就是说,如果一张PNG
图片能够被打开,那么它在文件的最后,必定存在IEND
标记。
回到刚才的Hex Compare,拉到最底部,于是发现:
没错。两张图片的末端都有IEND
标记。
也就是说,那张存在黑块的.png
文件,IO写入并没有问题。于是可以得出一个让人惊惶的结论:那台国产机的BitmapFactory
的底层处理有问题。
没错,就是这么坑。
解决方案
现在的问题很明确,BitmapFactory
中某些native
方法存在bug。那是不是所有的native
方法都有问题呢?
BitmapFactory.decodeStream(InputStream)
方法最终调用的是native
方法nativeDecodeStream(InputStream, byte[], Rect, Options)
。尝试绕开它试试看。
可否尝试将网络数据流保存到内存,随后再将其指向BitmapFactory
?答案是肯定的。我们尝试替换一部分代码。将此部分代码:
// 拿到输入流,此流即是图片资源本身
InputStream imputStream = conn.getInputStream();
// 指使Bitmap通过流获取数据
Bitmap bitmap = BitmapFactory.decodeStream(imputStream);
替换成:
// 拿到输入流,此流即是图片资源本身
InputStream imputStream = conn.getInputStream();
// 将所有InputStream写到byte数组当中
byte[] targetData = null;
byte[] bytePart = new byte[4096];
while (true) {
int readLength = imputStream.read(bytePart);
if (readLength == -1) {
break;
} else {
byte[] temp = new byte[readLength + (targetData == null ? 0 : targetData.length)];
if (targetData != null) {
System.arraycopy(targetData, 0, temp, 0, targetData.length);
System.arraycopy(bytePart, 0, temp, targetData.length, readLength);
} else {
System.arraycopy(bytePart, 0, temp, 0, readLength);
}
targetData = temp;
}
}
// 指使Bitmap通过byte数组获取数据
Bitmap bitmap = BitmapFactory.decodeByteArray(targetData, 0, targetData.length);
BitmapFactory.decodeByteArray(byte[], int, int)
方法最终调用了native
方法nativeDecodeByteArray(byte[], int, int, Options)
,与通过InputStream
处理所指向的native
方法不同。
经过测试,使用这种方法所保存的.png
文件不存在黑块问题。我们无法得知厂商ROM中对于这两种方法有什么差异对待,但至少可以明确:上文中提到的那台国产机子,通过InputStream
传递WebP
数据并存储为.png
图像这一过程存在可预知的bug。
至此,问题分析及解决方案阐述完毕。
原文地址http://www.jianshu.com/p/e5837a85e6cb
Android Bitmap转换WebP图片导致损坏的分析及解决方案的更多相关文章
- Android Bitmap转换WebPng图片导致损坏的分析及解决方案
出现问题的code!!! private void saveImage(String uri, String savePath) throws IOException { // 创建连接 HttpUR ...
- Android View转换成图片保存
package zhangphil.viewtoimage; import java.io.File;import java.io.FileOutputStream; import android.o ...
- android bitmap compress(图片压缩)
android bitmap compress android的照相功能随着手机硬件的发展,变得越来越强大,能够找出很高分辨率的图片. 有些场景中,需要照相并且上传到服务,但是由于图片的大小太大,那么 ...
- webp图片技术调研最终结论(完全真实数据可自行分析)
关于webp图片格式调研及测试 资料收集 什么是 WebP? WebP(发音 weppy),是一种支持有损压缩和无损压缩的图片文件格式,派生自图像编码格式 VP8.根据 Google 的测试,无损压缩 ...
- Android Bitmap 和 ByteArray的互相转换
Android Bitmap 和 ByteArray的互相转换 移动平台图像处理,需要将图像传给native处理,如何传递?将bitmap转换成一个 byte[] 方便传递也方便cpp代码直接处理图像 ...
- 【Android】Bitmap加载图片错误 java.lang.OutOfMemoryError: bitmap size exceeds VM budget
今天测试程序的时候出现下面的错误日志信息,程序当场挂掉 07-09 14:11:25.434: W/System.err(4890): java.lang.OutOfMemoryError: bitm ...
- Android bitmap图片处理
一.View转换为Bitmap 在Android中所有的控件都是View的直接子类或者间接子类,通过它们可以组成丰富的UI界面.在窗口显示的时候Android会把这些控件都加载到内存中 ...
- Android Bitmap与DrawAble与byte[]与InputStream之间的转换工具类【转】
package com.soai.imdemo; import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; ...
- Android,View转换bitmap,bitmap转换drawable
Android View转换Bitmap,Bitmap转换Drawable //测试设置bitmap View view1 = ViewGroup.inflate(context, R.layout. ...
随机推荐
- symfony 数据库中文乱码
这个问题 是由于编辑器没有设置utf8格式造成的,当然config里也要设置utf8 解决方法:编辑器设置utf8,重启 doctrine: dbal: driver: pdo_mysql host: ...
- 使用百度网盘实现自动备份VPS
http://ju.outofmemory.cn/entry/51536 经过轰轰烈烈的一轮网盘大战,百度网盘的容量已经接近无限(比如我的是3000多G ),而且百度网盘已经开放API,所以用来备份V ...
- [TypeScript] Use TypeScript’s never Type for Exhaustiveness Checking
TypeScript 2.0 introduced a new primitive type called never, the type of values that never occur. It ...
- C++虚函数表剖析
关键词:虚函数.虚表,虚表指针,动态绑定,多态 一.概述 为了实现C++的多态,C++使用了一种动态绑定的技术. 这个技术的核心是虚函数表(下文简称虚表).本文介绍虚函数表是怎样实现动态绑定的. 二. ...
- rsh 无秘钥登陆配置
/etc/hosts.equiv里的主机不须要提供password就能够訪问本机./etc/host.equiv 要和~/.rhosts文件连用. [root@web-htl2-01 ~]# cat ...
- jqury-validate表单验证
首先需要引入插件:jquery.validate.js这个插件. 然后对需要验证的表单实现js: $("#add-firewalls-form").validate({ submi ...
- 图像处理之基础---用Shader实现的YUV到RGB转换:使用3重纹理实现 .
上一篇中,我是用一个RGB格式的纹理来存储每一帧的画面,其中纹理为m_FrameWidth * m_FrameHeight大小,这样,在内存中,就必须要先对YUV的数据进行排序,然后才能当做RGB的数 ...
- 【Android基础】App签名与打包
签名的意义 1. 为了保证程序开发人员的合法 2. 防止部分人通过使用同样的Package Name(包名)来混淆替换已安装的程序 3. 保证我们每次公布的版本号的一致性(保证签名一致才干升级) 签名 ...
- Java:String和Date、Timestamp之间的转换【转】
原文地址:http://yunnick.iteye.com/blog/1074495 一.String与Date(java.util.Date)互转 1.1 String -> Date Str ...
- Aaron Swartz Rewriting Reddit中关于web.py的创建思路
这天才少年居然自杀了,哎 原文点这 So how should things work? The first principle is that code should be clear and si ...