前言:

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

(图片来自网络)

1.JPEG

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

(1)Baseline JPEG(标准型)

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

(图片来自网络)

(2)Progressive JPEG(渐进式)

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

(图片来自网络)


(图片来自网络)

2.解码

如何判断是否JPEG格式的图片呢?下面引用一段Glide框架的代码
//ImageHeaderParser.javaprivate 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

比较完整的数据结构如下

(图片来自Wiki)
https://en.wikipedia.org/wiki/JPEG

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函数我们可以轻松解决转换问题。

<?php$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 <inputfile> <outputfile>

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端做渐进式加载

热门文章

  • Glide-源码详解
  • 活用productFlavors
  • onTouch事件传递
  • 那些年我们解决滑动冲突时遇过的坑
  • 进程间通信–AIDL
  • 序列化–Serializable与Parcelable
  • 如何解决内存溢出以及内存泄漏
  • Okhttputils终极封装
  • FaceBook推出的调试神器
  • Android代码优化工具
  • Glide-入门教程
  • Glide-图片预处理(圆角,高斯模糊等)
  • Glide-图片的压缩
  • Glide-内存缓存与磁盘缓存
  • Glide-自定义缓存

渐进式加载-基础讲解相关推荐

  1. Android渐进式加载图片,渐进式加载 - 基础讲解

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

  2. Android插件化开发之动态加载基础之ClassLoader工作机制

    类加载器ClassLoader 早期使用过Eclipse等Java编写的软件的同学可能比较熟悉,Eclipse可以加载许多第三方的插件(或者叫扩展),这就是动态加载.这些插件大多是一些Jar包,而使用 ...

  3. 001:Mapbox GL加载基础的地图

    第001个 点击查看专栏目录 本示例的目的是介绍演示如何在vue+mapbox中加载最基础的 Mapbox GL地图 . 直接复制下面的 vue+mapbox源代码,操作2分钟即可运行实现效果 文章目 ...

  4. GameFramework篇:StarForce资源加载细节讲解(一:资源加载辅助器)

    前前后后看了一星期,才有了这篇博文,再次感叹,心急吃不了热豆腐. 在看这篇博文之前,建议先去了解一下Assetbundle和StreamingAsset和WWW和WebRequest这几个东西以及他们 ...

  5. GameFramework篇:StarForce资源加载细节讲解(二:ResourceComponent详解)

    上篇博客已经将编辑器和非编辑器资源加载区分开了,那我们这篇就来具体看看非编辑器模式下资源加载 进入InitResources函数,此时的ResourceManager已经是我们非编辑器模式下的资源器了 ...

  6. composer php中如何执行,php中composer如何实现类的自动加载(示例讲解)

    本篇文章给大家带来的内容是关于php中composer如何实现类的自动加载(示例讲解),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助. 我们在开发项目中会经常用到第三方的类库插件,但是 ...

  7. 【微信技术-微信小程序】------- 渐进式骨架屏(加载流)(第二篇)

    提示:看如下内容需要了解第一篇:简单入门(骨架屏(加载流) 简单入门(第一篇)) 目录 一.什么是渐进式骨架屏? 二.实现渐进式骨架屏 三.效果展示 下载示例代码地址: 渐进式骨架屏示例代码-下载 一 ...

  8. VUE项目中图片加载过大处理方式-渐进式方式-懒加载方式--附源码

    解决有两步: 1.采用渐进式加载:原图未加载完时显示比它内存小的模糊图: 2.采用懒加载:只加载可视区域的图片,即滚动到可视区域时再加载图片 vue-lazyload-img:VUE图片懒加载插件 v ...

  9. 【Android 应用开发】UI绘制流程 ( 生命周期机制 | 布局加载机制 | UI 绘制流程 | 布局测量 | 布局摆放 | 组件绘制 | 瀑布流布局案例 )

    文章目录 一. 博客相关资料 及 下载地址 1. 代码查看方法 ( ① 直接获取代码 | ② JAR 包替换 ) 2. 本博客涉及到的源码查看说明 二. Activity 生命周期回调机制 1. An ...

最新文章

  1. 如何让SiteMapPath使用指定路径指定名称的sitemap文件。
  2. [摘抄] Bezier曲线、B样条和NURBS
  3. f12 卡 谷歌浏览器_干货分享。关于谷歌浏览器的使用技巧,不需要插件
  4. Linux日志 系统日志及分析
  5. linux 磁盘资源管理以及IO
  6. 如何在 IIS 中设置 HTTPS 服务
  7. matlab 从 excel读取 日期_MATLAB批量修改文件名和选择性复制/剪切文件
  8. 计算机论文哪儿找,查找关于计算机毕业论文.docx
  9. React Native (一) react-native-video实现音乐播放器和进度条的功能
  10. 技术总监谈好的程序员如何写代码[转]
  11. PHP两种redirect
  12. 【学习】自定义view
  13. android各版本源码下载
  14. 计算机四级网络工程师总结
  15. vue 全年日历显示并且日期批量选择
  16. CentOS7 Tableau Server安装方法
  17. nyoj145 聪明的小柯
  18. Bazinga(HDU5510+KMP)
  19. tan x x的matlab求解,matlab画x=tan(x)
  20. 股市入门基础 :基本术语和概念的解读

热门文章

  1. Opencv中的convertTo
  2. MySQL——MySQL的flush
  3. sql 服务器停止运行,3.1.1 启动、停止、暂停和重新启动SQL Server服务
  4. vba访问服务器中txt文件,vba读取word内容 vba读取txt文件
  5. 请收起你的善良,越是心地善良的女人,通常命越差
  6. linux kobject-uevent(热插拔事件)
  7. sublime 使用浏览器不反应
  8. 微信搭建本地开发测试环境
  9. react-native报错-Unable to resolve module ‘react’
  10. iOS开发之常用第三方框架