当应用程序需要将位图图像保存到本地或发送到服务端时, 通常的方法是在发送数据前将图像通过PNG或JPEG编码。如果只是想保存位图图像,只要序列化BitmapData即可,将图像转换为JPEG/PNG是完全没有必要的。
BitmapData 转换为 ByteArray
获得BitmapData对应的字节数组, 所要做的只是调用getPixels()方法。getPixels()方法需要指定捕捉区域;最便捷的方法就是使用即将序列化的BitmapData的rect属性。
- // ActionScript 3.0
- // 假定“bitmapImage”是需要序列化的位图对象
- var bytes:ByteArray = bitmapImage.bitmapData.getPixels(bitmapImage.bitmapData.rect);
得到ByteArray对象后, 压缩:
- var bytes:ByteArray = bitmapImage.bitmapData.getPixels(bitmapImage.bitmapData.rect);
- bytes.compress();
位图尺寸(宽与高)
这样看来, 得到位图图像对应的ByteArray数据很容易 - 只要调用getPixel()方法即可.当然, 将ByteArray再构造为位图图像才能证明数据是有用的. 除像素数据外, 字节数组不能为位图图像指定尺寸.就是说你得把尺寸信息也要保存在字节数组里.其实只要保存高度或宽度即可, 因为已经知道了像素总数, 通过计算便能算出另一个.
下面的代码中,字节数组前4个字节保存BitmapData的宽度, 接下来再保存图像字节数组.
- var bytes:ByteArray = new ByteArray();
- bytes.writeUnsignedInt(bitmapImage.bitmapData.width);
- bytes.writeBytes(bitmapImage.bitmapData.getPixels(bitmapImage.bitmapData.rect));
- bytes.compress();
前面的工作完成后就可以使用常用的方法保存二进制数据了(发送给服务端脚本,AIR本地文件 API,SharedObject以及FP10 FileReference等等).这个例子中, 我们通过使用FileReference类的save()方法(需要Flash Player 10)将二进制数据保存到本地存储器中.由于Flash Player的安全措施,save()方法只有在用户交互 事件中才能够调用(例如 鼠标点击事件).因此需要新建一个 按钮并附加一个监听器, 在事件处理方法中调用save()方法.
- // ** 需要Flash Player 10以上版本 **
- function on_buttonClick(evt:MouseEvent):void
- {
- var bytes:ByteArray = new ByteArray();
- bytes.writeUnsignedInt(bitmapImage.bitmapData.width); // 保存图像宽度
- bytes.writeBytes(bitmapImage.bitmapData.getPixels(bitmapImage.bitmapData.rect)); //保存图像字节数组
- bytes.compress();
- new FileReference().save(bytes, "image.bmd"); // 默认文件名: "image.bmd"
- }
ByteArray 转换为 BitmapData
上面提到过, 我们要将保存的数据重构, 这样才能还原出原始位图图像.
首先, 通过URL Loader 加载文件:
- var ldr:URLLoader = new URLLoader();
- ldr.dataFormat = URLLoaderDataFormat.BINARY; // ** 这里一定要指定dataFormat为URLLoaderDataFormat.BINARY **
- ldr.addEventListener(Event.COMPLETE, on_fileLoad);
- ldr.addEventListener(IOErrorEvent.IO_ERROR, on_fileLoadError);
- ldr.load(new URLRequest(pathToBitmapDataFile));
- function on_fileLoad(evt:Event):void
- {
- if (evt.type == Event.COMPLETE)
- {
- var data:ByteArray = URLLoader(evt.target).data as ByteArray;
- if (data)
- {
- try
- {
- data.uncompress();
- }
- catch(e:Error)
- {
- }
- // 此时的数据已经是解压后的字节数组了
- // ... 处理数据 ...
- }
- }
- }
- // 数据解压后
- var width:int = data.readUnsignedInteger(); // 起始的4个字节
- // after data.uncompress()
- var height:int = ((data.length - 4) / 4) / width;
- // (data.length - 4) ** 去掉开始的4个字节,其余的便是位图的字节数组了 **
- // ((data.length - 4) / 4) ** 每个像素4个字节长, 所以要除以4得到总像素数 **
- // ((data.length - 4) / 4) / 宽度 ** 记住,因为是矩形才能这样计算出高度 **
得到尺寸后, 就可以使用setPixels()方法重构Bitmap对象了.
- var bmd:BitmapData = new BitmapData(width, height, true, 0); // 32位支持alpha通道的位图
- bmd.setPixels(bmd.rect, data); // 数据的position指向第5个字节了
- var bm:Bitmap = new Bitmap(bmd);
- addChild(bm);
以上方法展现了将BitmapData数据转换为ByteArray, 保存ByteArray, 然后再将已保存的ByteArray重新构造为BitmapData的整个过程.虽然基本目标是能够把位图图像保存到服务器/本地存储器, 但上述技巧放在其他情况中也是十分有用的.例如, 得到图像的ByteArray数据后, 可以将其发送(post)到服务器做进一步处理. 也可用来裁减外部的JPEG/PNG图像文件,去掉所有的JPEG/PNG编码中含有的元数据信息(meta information), 只留下原始(raw)图像数据(文件可能更小了).当然了, 最终的二进制文件不能做为JPEG/PNG打开了, 但 应用程序能够在运行时很容易的重构出相应的图像来.实际上,也可认为这是一种保护外部 图片不被盗链的好方法.