JS在浏览器中解析Base64编码图像


Posted in Javascript onFebruary 09, 2017

上一篇介绍中,我们将二进制文件(BLOB)保存为Base64编码的文本,这些文本可以内嵌在XML的标签中,因此二进制信息它可以随着XML文件被拷贝、下载而不用担心信息会缺失。这项技术也在email邮件中被广泛使用。

浏览器对Base64的支持

图像是最经常被使用的一种二进制文件。而现代的浏览器的进步日新月异,IE7,FireFox和其他浏览器为包括Base64在内各种编码的图像信息提供了很好的支持。因此图形信息可以以下面的形式呈现在页面中、

Java代码

<img src="data:image/gif;base64,R0lGODlhDwAPAKECAAAAzMzM///// 
wAAACwAAAAADwAPAAACIISPeQHsrZ5ModrLlN48CXF8m2iQ3YmmKqVlRtW4ML 
wWACH+H09wdGltaXplZCBieSBVbGVhZCBTbWFydFNhdmVyIQAAOw==" 
alt="Base64 encoded image" width="150" height="150"/>

这种data: URI的格式能把Base64(或其他数据)可以内嵌在image标签的属性当中(或者CSS中)。我们可以看到在大部分浏览器中的显示效果: 

JS在浏览器中解析Base64编码图像

这种做法有利有弊,好处是浏览器可以在一个连接中得到完成的页面内容,不好的地方时图像的大小会增加1/3。因此,这种内嵌的方法适合对小的图形元素比如图标、圆角等等进行处理,从而减少浏览器打开的连接数,但对大的照片、图片(量少而大)等等则不应该使用Base64编码以免影响下载速度。

为了得到刚才的Base64编码,我们将上一篇的Java修改成Struts Action,并借用了JIMI进行图形的读取和格式转换,Base64编码器则改为更普遍的Apache Commons组件,代码如下:

Java代码 

public class Base64ImageAction extends ActionSupport { 
  private final static String galleryName = "gallery"; 
 private static String parent = null; 
   private String encodeString = null; 
 public String getEncodeString() { 
  return encodeString; 
 } 
 public void setEncodeString(String encodeString) { 
  this.encodeString = encodeString; 
 } 
 private String getImageFullPath() { 
  parent = new File(this.getClass().getClassLoader().getResource( 
     File.separator).getPath()).getParent()+File.separator+"flag.jpg"; 
 } 
 public String execute() { 
  ByteArrayOutputStream output = new ByteArrayOutputStream(); 
  try { 
   JimiReader reader = Jimi.createJimiReader(this.getImageFullPath()); 
   Image image = reader.getImage(); 
   Jimi.putImage("image/png", image, output); 
   output.flush(); 
   output.close(); 
   this.encodeString = Base64.encodeBase64String(output.toByteArray()); 
  } catch (IOException e) { 
   e.printStackTrace(); 
  } catch (JimiException e) { 
   e.printStackTrace(); 
  } 
  return SUCCESS; 
 } 
}

对应的View端是个十分简单的Freemarker模板:

Html代码

<html> 
<head> 
<title>Hello,World</title> 
</head> 
<body> 
<img src="data:image/png;base64,${encodeString}" /> 
</body> 
</html>

处理古代浏览器

世界总是不是那么完美,尽管大部分现代浏览器对Base64的处理都十分完善,但是我们不能不考虑到一些“古老”的浏览器,而现在还是普遍使用的“古老”的浏览器,就当属IE6,在IE6里试图浏览上面的图片可能会得到一个红叉叉。我们不得不为IE6做一些特殊处理,利用下面的JavaScript,我们把Base64字串传回服务器端,重新解析成图片

Javascript代码 

// a regular expression to test for Base64 data 
var BASE64_DATA = /^data:.*;base64/i; 
// path to the PHP module that will decode the encoded data 
var base64Path = "/my/path/base64.php"; 
function fixBase64(img) { 
 // check the image source 
 if (BASE64_DATA.test(img.src)) { 
 // pass the data to the PHP routine 
 img.src = base64Path + "?" + img.src.slice(5); 
 } 
}; 
// fix images on page load 
onload = function() { 
 for (var i = 0; i < document.images.length; i++) { 
 fixBase64(document.images[i]); 
 } 
};

服务器端的Struts可以参考上面的例子做反向操作,具体从略。

更完美的方法

将Base64传回服务器解码是不错的IE6补丁,但是违背了我们的初衷,对IE6来说,浏览器连接数并未有任何减少。更直接的想法,是否能用Javascript直接在浏览器中,对Base64文本进行解码呢?我们构思的场景如下:服务器端先将图片转换成PNG格式以方便客户端进行处理,Base64编码之后,利用JSON将文本传递给浏览器客户端进行处理。

我们选择PNG图形格式是因为PNG已经俨然成为新的Web图形标准,它格式非常简单,可以很方便的用javascript进行处理而不需要借助浏览器的支持。我们知道javascript直接不能处理二进制数据,但是现在这不是个问题,服务器端已经准备好了Base64编码的文本数据,现在我们只需要一个javascript的Base64解析器,你可以在这里找到一个notmasteryet的Base64解析器。

现在PNG图形格式采用了DEFLATE作为唯一的压缩算法,该算法也广泛应用在ZIP,GZIP等压缩格式中。PNG图像格式文件(或者称为数据流)由一个8字节的PNG文件署名(PNG file signature)域和按照特定结构组织的3个以上的数据块(chunk)组成。

PNG定义了两种类型的数据块,一种是称为关键数据块(critical chunk),这是标准的数据块,另一种叫做辅助数据块(ancillary chunks),这是可选的数据块。关键数据块定义了4个标准数据块,其中图像数据块IDAT(image data chunk):它存储实际的数据, PNG总的数据流采用DEFLAT进行压缩。此外还擦用三角过滤“delta filters”来过滤每一行的像素的未压缩数据。DEFLAT和delta压缩在其他数据和文本处理中也被广泛应用。PNG格式你可以参考<a href="http://www.libpng.org/pub/png/spec/1.1/PNG-Contents.html">官方文档</a>。

很棒的,notmasteryet也为我们提供了一个DEFLAT解压器。

最后,我们把这些组合起来:

Html代码

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> 
<html xmlns="http://www.w3.org/1999/xhtml"> 
<head> 
 <title>Demo JavaScript PNG Viewer</title> 
 </head> 
<body onload="show(gravatar);"> 
<script src="../Source/Base64.js" type="text/javascript"></script> 
<script src="../Source/Deflate.js" type="text/javascript"></script> 
<script src="../Source/PNG.js" type="text/javascript"></script> 
<script type="text/javascript"> 
var gravatar = 'iVBORw0KGgoAAAANSUhEUgAAA.......数据从略......55CYII='; 
String.prototype.padRight = function(c, n){ 
 var txt = ''; 
 for(var i=0;i<n-this.length;i++) txt += c; 
 return txt + this; 
}; 
function show(data){ 
 var png = new PNG(data); 
 var img = document.getElementById('image'), limg = document.getElementById('largeimage'); 
 document.getElementById('nativeimage').src = 'data:image/png;base64,' + data; 
 img.innerHTML = ''; 
 limg.innerHTML = ''; 
 img.style.width = png.width + 'px'; 
 img.style.height = png.height + 'px'; 
 limg.style.width = (png.width * 3) + 'px'; 
 limg.style.width = (png.height * 3) + 'px'; 
 var line; 
 while(line = png.readLine()) 
 { 
  for (var x = 0; x < line.length; x++){ 
   var px = document.createElement('div'), px2 = document.createElement('div'); 
   px.className = px2.className = 'pixel'; 
   px.style.backgroundColor = px2.style.backgroundColor = '#' + line[x].toString(16).padRight('0', 6); 
   img.appendChild(px); 
   limg.appendChild(px2); 
  } 
 } 
} 
</script> 
<div id="image"></div> 
<div id="largeimage"></div> 
<img id="nativeimage" /> 
</body> 
</html>

以上所述是小编给大家介绍的JS在浏览器中解析Base64编码图像,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!

Javascript 相关文章推荐
javascript Object与Function使用
Jan 11 Javascript
JavaScript cookie的设置获取删除详解
Feb 11 Javascript
jQuery实现Div拖动+键盘控制综合效果的方法
Mar 10 Javascript
javascript实现显示和隐藏div方法汇总
Aug 14 Javascript
JS实现横向拉伸动感伸缩菜单效果代码
Sep 04 Javascript
基于javascript实现全国省市二级联动下拉选择菜单
Jan 28 Javascript
jQuery代码实现图片墙自动+手动淡入淡出切换效果
May 09 Javascript
js学习心得_一个简单的动画库封装tween.js
Jul 14 Javascript
js解决软键盘遮挡输入框的问题分享
Dec 19 Javascript
vue中mint-ui的使用方法
Apr 04 Javascript
vue-router 2.0 跳转之router.push()用法说明
Aug 12 Javascript
JS PHP字符串截取函数实现原理解析
Aug 29 Javascript
jQuery实现在新增加的元素上添加事件方法案例分析
Feb 09 #Javascript
jquery滚动条插件slimScroll使用方法
Feb 09 #Javascript
JS回调函数简单用法示例
Feb 09 #Javascript
Bootstrap列表组学习使用
Feb 09 #Javascript
jQuery实现获取隐藏div高度的方法示例
Feb 09 #Javascript
Bootstrap进度条学习使用
Feb 09 #Javascript
详解浏览器渲染页面过程
Feb 09 #Javascript
You might like
PHP按行读取、处理较大CSV文件的代码实例
2014/04/09 PHP
php格式化金额函数分享
2015/02/02 PHP
Yii中表单用法实例详解
2016/01/05 PHP
PHP基于单例模式实现的mysql类
2016/01/09 PHP
php实现学生管理系统
2020/03/21 PHP
centos下file_put_contents()无法写入文件的原因及解决方法
2017/04/01 PHP
thinkphp5.0自定义验证规则使用方法
2017/11/16 PHP
ajax+php实现无刷新验证手机号的实例
2017/12/22 PHP
学习YUI.Ext 第六天--关于树TreePanel(Part 2异步获取节点)
2007/03/10 Javascript
JavaScript 解析读取XML文档 实例代码
2009/07/07 Javascript
用unescape反编码得出汉字示例
2014/04/24 Javascript
jQuery中insertBefore()方法用法实例
2015/01/08 Javascript
jQuery过滤选择器详解
2015/01/13 Javascript
深入浅析JavaScript系列(13):This? Yes,this!
2016/01/05 Javascript
jQuery webuploader分片上传大文件
2016/11/07 Javascript
AngularJS使用ng-app自动加载bootstrap框架问题分析
2017/01/04 Javascript
解决vue中使用Axios调用接口时出现的ie数据处理问题
2018/08/13 Javascript
React注册倒计时功能的实现
2018/09/06 Javascript
JS轮播图的实现方法2
2020/08/25 Javascript
使用rpclib进行Python网络编程时的注释问题
2015/05/06 Python
django1.11.1 models 数据库同步方法
2018/05/30 Python
Python中flatten( )函数及函数用法详解
2018/11/02 Python
Python3.5 Pandas模块之DataFrame用法实例分析
2019/04/23 Python
python实现银行实战系统
2020/02/26 Python
canvas三角函数模拟水波效果的示例代码
2018/07/03 HTML / CSS
HTML5之SVG 2D入门13—svg对决canvas及长处和适用场景分析
2013/01/30 HTML / CSS
Max&Co官网:意大利年轻女性时尚品牌
2017/05/16 全球购物
意大利时尚精品店:Nugnes 1920
2020/02/10 全球购物
一些Solaris面试题
2013/03/22 面试题
一份软件工程师的面试试题
2016/02/01 面试题
计算机专业毕业生自荐信
2013/12/31 职场文书
《我的伯父鲁迅先生》教学反思
2014/02/12 职场文书
制药工程专业职业生涯规划范文
2014/03/10 职场文书
2014年后备干部工作总结
2014/12/08 职场文书
2016年企业先进员工事迹材料
2016/02/25 职场文书
TypeScript 使用 Tuple Union 声明函数重载
2022/04/07 Javascript