前言:

我们在PC端用浏览器看图片的时候,经常是先看到一张模糊图,然后再渐渐的变得清晰,这种情况在看漫画的时候尤其常见(模糊图如下),这种效果就叫做渐进式加载.渐进式加载能够大大的提升体验感,我们先来了解一下渐进式加载的原理.

图片来自网络

(图片来自网络)

1.JPEG

要做到渐进式加载,我们的图片需要是JPEG格式,而JPEG格式的图片又分为两种,我们要做到渐进式加载的话,需要的是Progressive JPEG.

(1)Baseline JPEG(标准型)

这种格式的图片在保存信息的时候,是从上往下,将每一行的数据顺序的保存起来的,所以读一部分就展示的话,那么效果就会像是从上往下一点一点展示.

(图片来自网络)

(2)Progressive JPEG(渐进式)

这种格式的图片在保存信息的时候,是一帧一帧的存储的,如果逐帧逐帧的读的话,就会先看到模糊图,然后一点一点变清晰

(图片来自网络)

(图片来自网络)

2.解码

如何判断是否JPEG格式的图片呢?下面引用一段Glide框架的代码

//ImageHeaderParser.java

private static final int EXIF_MAGIC_NUMBER = 0xFFD8;

// JPEG.

if (firstTwoBytes == EXIF_MAGIC_NUMBER) {

return JPEG;

}复制代码

我们可以看出,JPEG是以FFD8开头的

其实JPEG是以FFD8开头,FFD9结尾,FFDA代表一个帧的开头

FFD8 ... FFDA ... FFDA ... FFDA ... FFD9复制代码

Baseline JPEG 里面只有一个FFDA

Progressive JPEG 里面含有多个FFDA

比较完整的数据结构如下

3.如何保存或者转换成JPEG

(以下转换方法来自网络,由于非java代码,所以没有做验证,特此说明一下)

1、PhotoShop

在photoshop中有“存储为web所用格式”,打开后选择“连续”就是渐进式JPEG。

2、Linux

检测是否为progressive jpeg : identify -verbose filename.jpg | grep Interlace(如果输出 None 说明不是progressive jpeg;如果输出 Plane 说明是 progressive jpeg。)

将basic jpeg转换成progressive jpeg:> convert infile.jpg -interlace Plane outfile.jpg

3、PHP

使用imageinterlace和imagejpeg函数我们可以轻松解决转换问题。

$im = imagecreatefromjpeg('pic.jpg');

imageinterlace($im, 1);

imagejpeg($im, './php_interlaced.jpg', 100);

imagedestroy($im);

?>复制代码

4、Python

import PIL

from exceptions import IOError

img = PIL.Image.open("c:\\users\\biaodianfu\\pictures\\in.jpg")

destination = "c:\\users\\biaodianfu\\pictures\\test.jpeg"

try:

img.save(destination, "JPEG", quality=80, optimize=True, progressive=True)

except IOError:

PIL.ImageFile.MAXBLOCK = img.size[0] * img.size[1]

img.save(destination, "JPEG", quality=80, optimize=True, progressive=True)复制代码

5、jpegtran

jpegtran -copy none -progressive 复制代码

6、C

using (Image source = Image.FromFile(@"D:\temp\test2.jpg")) {

ImageCodecInfo codec = ImageCodecInfo.GetImageEncoders().First(c => c.MimeType == "image/jpeg");

EncoderParameters parameters = new EncoderParameters(3);

parameters.Param[0] = new EncoderParameter(System.Drawing.Imaging.Encoder.Quality, 100L);

parameters.Param[1] = new EncoderParameter(System.Drawing.Imaging.Encoder.ScanMethod, (int)EncoderValue.ScanMethodInterlaced);

parameters.Param[2] = new EncoderParameter(System.Drawing.Imaging.Encoder.RenderMethod, (int)EncoderValue.RenderProgressive);

source.Save(@"D:\temp\saved.jpg", codec, parameters);

}复制代码

4.效果

明白了渐进式加载的原理后,我们就能想办法在app端也做到渐进式加载的效果了.

(大概就是判断是否JPEG图片,然后根据每一帧的节点来判断并决定是否需要加载)

下面展示一下效果图

(1)原图

(Progressive JPEG的图一打水印就变成Baseline JPEG,应该是CSDN打水印保存的时候处理了)

(2)解码到第一个FFDA与第二个FFDA的中间

(3)刚好解码到第二个FFDA

(4)解码到第五个FFDA

需要看图片二进制结构的,可以下载一些工具(如hex-editor-neo)

hex-editor-neo下载

在后面的文章里面我们将具体讲解如何在app端做渐进式加载

热门文章

Android渐进式加载图片,渐进式加载 - 基础讲解相关推荐

  1. Android 数据库中读取图片名称加载相应的资源

    android的开发中会遇到这么一种需求,就是把图片存储到本地,每次加载的时候通过保存的图片的名称加载图片,或者利用请求到的/存储数据库存储的名称去加载图片 . 开始我把数据放在了assets中,开始 ...

  2. html图片懒加载,图片懒加载原理及实现

    原理: 先将img标签的src链接设为同一张图片(比如空白图片),然后给img标签设置自定义属性(比如 data-src),然后将真正的图片地址存储在data-src中,当JS监听到该图片元素进入可视 ...

  3. html和css的图片怎么加,css图片怎么加链接?

    css可以通过在图片前加标签实现为图片加链接,语法:.使用此方法引入图片即可为图片添加链接. 设定图片超链接,用css样式来实现,其实就是给这个图片的块添加超链接,我们可以通过在这个块的前面添加一个a ...

  4. Android下WebView的图片适配加载与文字自动换行处理

    相信各位在Android开发中使用WebView加载页面的时候,或多或少会遇到一些网页适配的问题,例如图片过大或者是单行文字过长, 导致网页可以左右拖动,使自己加载出来的网页显得非常的low,下面介绍 ...

  5. RecycleView中使用Glide加载图片防止加载错乱

    前言 使用recyclerView时遇到了图片错位的问题,这个问题网上已经讨论的很成熟,谨以此文章做个总结. 问题产生原因 根本原因: 因为有ViewHolder的重用机制,每一个item在移出屏幕后 ...

  6. js 加载图片无法加载成功,提示:net::ERR_BLOCKED_BY_CLIENT

    1.浏览器有禁止广告的插件,关掉就可以了. 2.插件图片命名是否 带有广告,ad,等词,如果有,删掉就可以.

  7. 使用selenium设置firefox不加载图片增加加载速度提高爬虫效率

    from selenium import webdriver options=webdriver.FirefoxProfile() options.set_preference('permission ...

  8. el-image加载图片初始化加载失败问题

    // file.url = URL.createObjectURL(res.data); // 使用this.$set()添加属性,解决初始化el-image的src获取不到url的问题this.$s ...

  9. android webview 图片异步加载,Webview 图片异步加载及bug解决

    在项目中新闻显示需要使用图片的异步加载,并且在加载图片前和加载图片失败时显示默认图片. 关于webview中图片的异步加载的思路如下: 方案一:由前端同学全权负责(使用js实现异步加载) 方案二:an ...

最新文章

  1. 第 127 章 Piranha - Cluster administation tools
  2. ThreadPoolExecutor线程池,shutdown和shutdownNow关闭线程池方式对比,以及确保线程池能够彻底关闭的一种方式
  3. 在Windows2003server上安装msn9.0以上版本的解决办法
  4. 二次元福利适应个人引导页
  5. 7月26日见!华为Mate 20 X 5G正式官宣:国内首款5G双模手机
  6. linux释放内存后设备起不来,Linux-Memory小记
  7. java程序员目标_Java程序员的目标,你都达到了多少条?
  8. HDU 1213 How Many Tables 并查集 水~
  9. 从stm32转向Linux,STM32MP1Distrib
  10. 用计算机绘制函数图像结题报告,《几何画板》在高中数学教学中的应用小课题结题报告...
  11. vs2008中caption属性如何设置为中文
  12. Wow~70G上市公司定期报告数据集!
  13. 识别文字的软件哪个好呢?不如试试这几款
  14. 手机游戏显示服务器异常,手机玩游戏是云服务器异常
  15. golang数据结构初探之字典map
  16. luogu1968 美元汇率
  17. IMX6UL eMMC命令分析
  18. 表单提交-form,快速取值
  19. yesry 通过桥接器,Java调用任何API或DLL
  20. docker和docker-compose部署

热门文章

  1. 2018届各大互联网公司校招薪资曝光汇总
  2. go golang 中获取string 长度
  3. 2022年湖南省自考考试学前儿童游戏指导练习题及答案
  4. 手机电视与奥运将擦肩而过
  5. 2023我的第一个个人软件作品 无忧桶装水配送管理系统 出来了
  6. Codeforces Round #105 D. Bag of mice 概率dp
  7. python身份证号判断性别。1.判断身份证长度是否正确 2.判断性别
  8. 微信小程序之from表单提交数据到PHP后台
  9. 判断对象数组中是否存在重复项
  10. 纲举目张:打通MySQL架构和业务的任督二脉