Chinaunix首页 | 论坛 | 博客
  • 博客访问: 818860
  • 博文数量: 62
  • 博客积分: 526
  • 博客等级: 二等列兵
  • 技术积分: 2078
  • 用 户 组: 普通用户
  • 注册时间: 2012-02-04 20:41
个人简介

博客迁移至 freefe.cc

文章分类

全部博文(62)

分类: JavaScript

2014-03-25 22:52:29

    

    好吧,本以为node中的Buffer只是nodeJS中挺小的一块,仔细的翻阅了一些资料之后才发现,冰川总是将其巨大的屁股藏在海平面以下的,这次也是主要简单的讲一下关于Buffer比较浅的一些东西(针对Node初学者啦~)


Buffer 是什么?


    对于JavaScript来说,对Unicode编码的数据很容易处理的但是对于二进制数据就没什么处理方法了。但是在一些TCP数据流或者在操作一些文件数据流的时候,字节流的处理方法还是必须的。nodeJS中便出了处理的策略方法~提供了与String类似对等的全局构造函数Buffer(与其说与String对等,还不如说与Array类似但是有些不同方面还是需要注意),全局那么自然就不要每次的require了。Buffer则node中储存二进制数据的中介者。

创建Buffer实例的方式:

    new Buffer( size );
    和数组类似,给定所需要创建Buffer对象的大小。



    new Buffer( array );
    给定二进制数据的一个数组形式排列



    new Buffer( str, [ encoding ] );
    给与某一特定数据类型数据,以及其编码类型(默认均为utf8)



   

以及还有一些操作Buffer实例的方法:


    一个Buffer的形式非常类似于一个整数数组,与字符串还是有一定的区别的。
    JS程序员都比较熟悉,对于字符串来说可以视为可读的,每次的修改,复制都会得到一个新的字符串,不会影响之前的字符串,就如有一个字符串a,将 var b = a;  a 和 b 是各会拥有独立的一份字符串数据,相互独立,与对象类似于指针的形式不同(按值传递和按址传递)。
    不过对于字符串的兄弟Buffer来说,作为对象的Buffer含有的Array的血更多一点。



    但是对于Buffer的一些方法进行使用的时候,还是需要注意其和数组一些不一致的地方




    在将创建的Buffer截取一段之后,对返回的Buffer段进行修改,还是会影响原Buffer,两者仍是关联的。这类似于一个储存一个个对象的数组,Buffer这个类数组中每一个元素非原始值,而是引用值。

    对于希望复制一份独立的Buffer拷贝,Buffer提供了专门的方法
    Buffer.copy(targetBuffer, [targetStart], [sourceStart], [sourceEnd]);

    相对来说并没有直接的直接拷贝副本的方法,只能新建一个长度相等的Buffer,然后在原Buffer上调用copy方法,参数中还可以设置copy的启事与结束位置等。




    对于更多的信息可以查看node的Buffer文档

什么时候才需要使用Buffer


    好了,以上就是对Buffer的简单介绍,其实那都是虚的,什么时候才需要使用Buffer呢?

    其实之前已经说过在Node中,许多地方的数据流均是使用的Buffer类型,以便于来处理一些二进制数据(比如读取文件数据,http请求中的post传递的数据等)。 


  1. var fs = require('fs');
  2. var rs = fs.createReadStream('chinese.md');

  3. rs.on("data", function (chunk){
  4.     console.log( Buffer.isBuffer( chunk ) )
  5. });




    从上面可知其均为 Buffer 类型片段,那么在一些情况下也会使得我们遇到一些隐藏的问题:

  1. var fs = require('fs');
  2. var rs = fs.createReadStream('chinese.md' , { highWaterMark: 5 } );
  3. var data = '';
  4. rs.on("data", function (chunk){
  5.     data += chunk;
  6. });
  7. rs.on("end", function () {
  8.     console.log(data);
  9. });



    会出现乱码现象,无法正常显示。



    一个中文字占了3个Buffer单位,但是在数据流分块的时候将一个中文的3个标识分开,再做 toString的时候救会发生出现这种乱码的情况。解决方式就是在拿到的分块数据千万不能直接进行类似toString式转义,将每段Buffer保存,最后合并成一个大的Buffer后在进行转义:

  1. var fs = require('fs');
  2. var rs = fs.createReadStream('chinese.md', { highWaterMark: 5 } );
  3. var dataArr = [], len = 0, data;
  4. rs.on("data", function (chunk){
  5.     dataArr.push(chunk);
  6.     len += chunk.length;
  7. });
  8. rs.on("end", function () {
  9.     data = Buffer.concat( dataArr, len ).toString();
  10.     console.log(data);
  11. });



上述实例中我们可能还发现一个问题,将 { highWaterMark: 5 } 参数去掉后其实两个均没有问题,原因是因为默认的stream中切割data块单位是8K,所以可以在一个单位内容下这个小实例,所以没什么问题。一旦说数据很大,那么也会有相应的问题了。(之前一直拿txt测试了好久,发现怎么也不对,发现是编码方式问题)。

顺便说一下在拼接性能方面,其实Buffer与String相比并不是很差:

  1. var buf = new Buffer('this is text concat test'),
  2.  str = 'this is text concat test';
  3. console.time('buffer concat test');
  4. var list = [];
  5. var len= 100000 * buf.length;
  6. for(var i=0;i<100000;i++){
  7.     list.push(buf);
  8.     len += buf.length;
  9. }
  10. var s1 = Buffer.concat(list, len).toString();
  11. console.timeEnd('buffer concat test');
  12. console.time('string concat test');
  13. var list = [];
  14. for (var i = 100000; i >= 0; i--) {
  15.   list.push(str);
  16. }
  17. var s2 = list.join('');
  18. console.timeEnd('string concat test');




好吧,我承认写的有点不耐烦了,那么就这样吧,相当于简单的做个标签,有需要再查吧~ 哦吼吼~
阅读(15607) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~