11RIA 闪客社区 - 最赞 Animate Flash 论坛

搜索
查看: 2104|回复: 2
上一主题 下一主题

[AS 开发 & 技巧] BitmapData 转换为 ByteArray

[复制链接] TA的其它主题
发表于 2018-12-4 14:01:25 | 显示全部楼层 |阅读模式

【游客模式】——注册会员,加入11RIA 闪客社区吧!一起见证Flash的再次辉煌……

您需要 登录 才可以下载或查看,没有帐号?立即注册

x
本帖最后由 TKCB 于 2018-12-4 15:15 编辑

BitmapData 转换为 ByteArray

获得BitmapData对应的字节数组, 所要做的只是调用getPixels()方法。getPixels()方法需要指定捕捉区域;最便捷的方法就是使用即将序列化的BitmapData的rect属性。

[Actionscript3] 纯文本查看 复制代码
// ActionScript 3.0
// 假定“bitmapImage”是需要序列化的位图对象
var bytes:ByteArray = bitmapImage.bitmapData.getPixels(bitmapImage.bitmapData.rect);


这个方法会返回一个ByteArray对象,BitmapData的每个像素对应ByteArray对象中的一个4字节的无符号整型。这意味着如果是20x20的位图图像, 对应的ByteArray对象在压缩前有1600个字节(20x20x4=1600)
得到ByteArray对象后, 压缩:

[Actionscript3] 纯文本查看 复制代码
var bytes:ByteArray = bitmapImage.bitmapData.getPixels(bitmapImage.bitmapData.rect);
bytes.compress();


得到了位图图像无损压缩的二进制数据了.


位图尺寸(宽与高)

这样看来, 得到位图图像对应的ByteArray数据很容易 - 只要调用getPixel()方法即可.当然, 将ByteArray再构造为位图图像才能证明数据是有用的. 除像素数据外, 字节数组不能为位图图像指定尺寸.就是说你得把尺寸信息也要保存在字节数组里.其实只要保存高度或宽度即可, 因为已经知道了像素总数, 通过计算便能算出另一个.

下面的代码中,字节数组前4个字节保存BitmapData的宽度, 接下来再保存图像字节数组.

[Actionscript3] 纯文本查看 复制代码
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()方法.

[Actionscript3] 纯文本查看 复制代码
// ** 需要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"
}


文件可任意命名.上面的例子中, 我使用了".bmd"(BitmapData)做为文件扩展名,不过这只是一个自己想出的文件类型.最终保存的文件无有效MIME的, 不会当作已知的文件类型运行 - 这是我们自定义的二进制数据格式文件, 仅仅是用来保存图像数据, 方便以后我们的程序重用.

ByteArray 转换为 BitmapData

上面提到过, 我们要将保存的数据重构, 这样才能还原出原始位图图像.
首先, 通过URLLoader加载文件:

[Actionscript3] 纯文本查看 复制代码
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));


事件处理方法on_fileLoad:

[Actionscript3] 纯文本查看 复制代码
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)
                        {
                        }
                        // 此时的数据已经是解压后的字节数组了
                        // ... 处理数据 ...
                }
        }
}


现在我们来取出位图图像的尺寸. 还记得之前我们在二进制数据的头4个字节保存了宽度值吧.

[Actionscript3] 纯文本查看 复制代码
// 数据解压后
var width:int = data.readUnsignedInteger(); // 起始的4个字节


得到高度:

[Actionscript3] 纯文本查看 复制代码
// 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对象了.

[Actionscript3] 纯文本查看 复制代码
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打开了, 但应用程序能够在运行时很容易的重构出相应的图像来.实际上,也可认为这是一种保护外部图片不被盗链的好方法.


评分

参与人数 2银子 +132 金子 +1 贡献 +1 收起 理由
TKCB + 66 + 1 + 1 稍微调整下格式吧,我已经帮你调整了下.
最后一批aser + 66 T大來加分~~

查看全部评分

 楼主| 发表于 2018-12-4 14:04:10 | 显示全部楼层
这个还是挺有用的,适合新手,关键这里还有新手嘛
回复

使用道具 举报

发表于 2018-12-4 14:18:29 | 显示全部楼层
有用的
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

关闭

站长推荐 上一条 /1 下一条

感谢所有支持论坛的朋友:下面展示最新的5位赞助和充值的朋友……更多赞助和充值朋友的信息,请查看:永远的感谢名单

SGlW(66139)、 anghuo(841)、 whdsyes(255)、 longxia(60904)、 囫囵吞澡(58054)

下面展示总排行榜的前3名(T1-T3)和今年排行榜的前3名的朋友(C1-C3)……更多信息,请查看:总排行榜今年排行榜

T1. fhqu1462(969)、 T2. lwlpluto(14232)、 T3. 1367926921(962)  |  C1. anghuo(147)、 C2. fdisker(27945)、 C3. 囫囵吞澡(58054)



快速回复 返回顶部 返回列表