Nodejs 中的 Buffer 類的創(chuàng)建與基本使用
前言
JavaScript 對(duì)于字符串(string)的操作十分友好,無(wú)論是寬字節(jié)字符串還是單字節(jié)字符串,都被認(rèn)為是一個(gè)字符串。
console.log("你好,世界!".length); // 6
console.log("hello,world!".length); // 12
console.log("\u00cc".length); // 1
作為對(duì)比,Rust 中的字符串則相對(duì)難以理解:
let str = String::from("你好,世界!");
println!("{}", str.len()); // 18
這是因?yàn)?Rust 中的 String 類型本身就是基于數(shù)組 vec 進(jìn)行的封裝,數(shù)組每個(gè)元素都是一個(gè) u8 類型的元素,而 JavaScript 的 String 類型的抽象程度要更為高。這符合這兩種語(yǔ)言的應(yīng)用面(一個(gè)作為系統(tǒng)編程語(yǔ)言,一個(gè)為腳本語(yǔ)言)。
而這種高抽象層次,在 Nodejs 拓展了 JavaScript 的應(yīng)用面之后,就顯得有些力不從心了。在 Nodejs 中,應(yīng)用需要處理網(wǎng)絡(luò)協(xié)議、操作數(shù)據(jù)庫(kù)、處理圖片、接收上傳文件等,在網(wǎng)絡(luò)流和文件的操作中,還要處理大量二進(jìn)制數(shù)據(jù)。 JavaScript 原有的字符串遠(yuǎn)遠(yuǎn)不能滿足這些需求,于是 Buffer 應(yīng)運(yùn)而生。
Buffer 結(jié)構(gòu)
Buffer 是一個(gè)像 Array 的對(duì)象,但它主要用于操作字節(jié)。
底層實(shí)現(xiàn)
Buffer 是一個(gè) JavaScript 與 C++ 結(jié)合的模塊,它將性能相關(guān)部分用 C++ 實(shí)現(xiàn),將非性能相關(guān)的部分用 JavaScript 實(shí)現(xiàn):

?? Buffer所占用的內(nèi)存不是通過(guò)V8分配的,屬于堆外內(nèi)存,這涉及V8內(nèi)存分配和垃圾回收機(jī)制。
?? Node 在進(jìn)程啟動(dòng)時(shí)就加載了 Buffer 類,并將其放在全局對(duì)象(global)上。你無(wú)需通過(guò) require 導(dǎo)入。
Buffer 對(duì)象
Buffer 對(duì)象類似于 Rust 中的 String 類型,它的元素為無(wú)符號(hào)8位二進(jìn)制數(shù),即0到255的數(shù)值:
console.log(new Buffer.from("hello,world!", "utf-8"));
// <Buffer 68 65 6c 6c 6f 2c 77 6f 72 6c 64 21>
?? 在 UTF-8 中,漢字一般占用 3 個(gè)元素,字母和半角標(biāo)點(diǎn)符號(hào)占用 1 個(gè)元素。
類似于字符串,你也可以使用 length 查看 Buffer 的長(zhǎng)度:
console.log(new Buffer.from("你好,世界!", "utf-8").length); // 14
你可以使用 alloc() 或 allocUnsafe() 創(chuàng)建一個(gè)指定長(zhǎng)度的 Buffer 對(duì)象:
const buf = new Buffer.alloc(100); const buf = new Buffer.allocUnsafe(100); // 創(chuàng)建一個(gè)長(zhǎng)為100的Buffer
如果賦超過(guò)0~255的值,則會(huì)發(fā)生數(shù)值溢出:
const buf = new Buffer.alloc(100); buf[20] = -100; buf[30] = 266; buf[40] = 3.1415; console.log(buf[20], buf[30], buf[40]); // 156 10 3
具體原理涉及計(jì)算機(jī)存儲(chǔ)數(shù)值的方法,簡(jiǎn)單來(lái)說(shuō):
- 給元素的賦值如果小于0,就將該值逐次加256,直到得到一個(gè)0到255之間的整數(shù)。
- 如果得到的數(shù)值大于255,就逐次減256,直到得到0~255區(qū)間內(nèi)的數(shù)值。
- 如果是小數(shù),舍棄小數(shù)部分,只保留整數(shù)部分。
?? 上面提到的 Buffer 對(duì)象都是 JavaScript 層面的,能夠被 V8 的垃圾回收標(biāo)記回收。但是其內(nèi)部的parent 指針指向的 SlowBuffer 對(duì)象卻來(lái)自于 Nodejs 自身 C++ 中的定義,是 C++ 層面上的 Buffer 對(duì)象,所用內(nèi)存不在 V8 的堆中,屬于堆外內(nèi)存。
Buffer 轉(zhuǎn)換
Buffer對(duì)象可以與字符串之間相互轉(zhuǎn)換。目前支持包括 ASCII、utf-8、base64、Binary 等多種字符串編碼類型。
字符串轉(zhuǎn)Buffer
通過(guò)構(gòu)造函數(shù):
new Buffer.from(str, [encoding]);
一個(gè)Buffer對(duì)象內(nèi)部可以存儲(chǔ)不同編碼類型的字符串轉(zhuǎn)碼的值,調(diào)用write()方法可以實(shí)現(xiàn)該目的:
write(string, [offset], [length], [encoding]);
Buffer轉(zhuǎn)字符串
Buffer對(duì)象的toString()可以將Buffer對(duì)象轉(zhuǎn)換為字符串:
toString([encoding], [startIndex], [endIndex])
可以設(shè)置encoding(默認(rèn)為UTF-8)、start、end這3個(gè)參數(shù)實(shí)現(xiàn)整體或局部的轉(zhuǎn)換。如果Buffer對(duì)象由多種編碼寫入,就需要在局部指定不同的編碼,才能轉(zhuǎn)換回正常的編碼。
?? Nodejs 內(nèi)置的 Buffer 支持的字符串編碼有限,如果想要實(shí)現(xiàn)與 GBK、GB2312 的轉(zhuǎn)換,請(qǐng)?jiān)谏鐓^(qū)尋找對(duì)應(yīng)包。
總結(jié)
講完了 Nodejs 中的 Buffer 類的創(chuàng)建與基本使用,接下來(lái)我們將介紹 Buffer 的應(yīng)用。Buffer在文件I/O和網(wǎng)絡(luò)I/O中運(yùn)用廣泛,尤其在網(wǎng)絡(luò)傳輸中,更多關(guān)于Nodejs Buffer類創(chuàng)建使用的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
node.js使用redis儲(chǔ)存session的方法
這篇文章主要介紹了node.js使用redis儲(chǔ)存session的方法,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2018-09-09
在Node.js中設(shè)置響應(yīng)的MIME類型的代碼詳解
在 Node.js 中設(shè)置響應(yīng)的 MIME 類型是為了讓瀏覽器正確解析服務(wù)器返回的內(nèi)容,比如 HTML、CSS、圖片、JSON 等,我們通常通過(guò)設(shè)置響應(yīng)頭中的 Content-Type 字段來(lái)完成,本文就給大家詳細(xì)介紹了在Node.js中設(shè)置響應(yīng)的MIME類型的方法,需要的朋友可以參考下2025-04-04
React和Node.js快速上傳進(jìn)度條功能實(shí)現(xiàn)
這篇文章主要為大家介紹了React和Node.js快速上傳進(jìn)度條功能實(shí)現(xiàn)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-03-03
Node.js?中常用內(nèi)置模塊(path?路徑模塊)
這篇文章主要介紹了Node.js?中常用內(nèi)置模塊(path?路徑模塊),文章圍繞主題展開詳細(xì)的相關(guān)介紹,具有一定的參考價(jià)值,感興趣的朋友可以參考一下2022-09-09
node實(shí)現(xiàn)將json轉(zhuǎn)為excel
平時(shí)我們寫代碼處理的數(shù)據(jù)格式一般都是json格式的數(shù)據(jù),但有時(shí)候我們也需要將數(shù)據(jù)轉(zhuǎn)為excel格式進(jìn)行保存或分享,所以下面我們就來(lái)學(xué)習(xí)一下如何通過(guò)node實(shí)現(xiàn)json轉(zhuǎn)excel吧2024-11-11
nodejs動(dòng)態(tài)創(chuàng)建二維碼的方法
這篇文章主要為大家詳細(xì)介紹了nodejs動(dòng)態(tài)創(chuàng)建二維碼的方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-08-08
win10環(huán)境使用nvm安裝多版本nodejs并配置環(huán)境變量的完整步驟
這篇文章主要給大家介紹了關(guān)于win10環(huán)境使用nvm安裝多版本nodejs并配置環(huán)境變量的相關(guān)資料,需要的朋友可以參考下2024-01-01
node.js解決全局安裝pnpm后無(wú)法使用的問(wèn)題
在全局安裝pnpm后,如果出現(xiàn)無(wú)法使用的問(wèn)題,一般是由于沒(méi)有添加系統(tǒng)變量導(dǎo)致的,本文就來(lái)介紹一下node.js解決全局安裝pnpm后無(wú)法使用的問(wèn)題,感興趣的可以了解一下2024-10-10

