使用Node.js怎么從字符串生成文件流?很多新手對此不是很清楚,為了幫助大家解決這個難題,下面小編將為大家詳細(xì)講解,有這方面需求的人可以來學(xué)習(xí)下,希望你能有所收獲。
網(wǎng)站建設(shè)哪家好,找創(chuàng)新互聯(lián)建站!專注于網(wǎng)頁設(shè)計、網(wǎng)站建設(shè)、微信開發(fā)、重慶小程序開發(fā)、集團(tuán)企業(yè)網(wǎng)站建設(shè)等服務(wù)項目。為回饋新老客戶創(chuàng)新互聯(lián)還提供了施秉免費建站歡迎大家使用!
var FormData = require('form-data'); var fs = require('fs'); var form = new FormData(); form.append('my_file', fs.createReadStream('/foo/bar.jpg')); form.submit('example.org/upload', function(err, res) { console.log(res.statusCode); });
不想寫物理文件的話,可以這樣做:
const FormData = require('form-data'); const filename = 'my-file.txt'; const content = 'balalalalala...變身'; const formData = new FormData(); // 1.先將字符串轉(zhuǎn)換成Buffer const fileContent = Buffer.from(content); // 2.補(bǔ)上文件meta信息 formData.append('file', fileContent, { filename, contentType: 'text/plain', knownLength: fileContent.byteLength });
也就是說,文件流除了能夠提供數(shù)據(jù)外,還具有一些 meta 信息,如文件名、文件路徑等 ,而這些信息是普通 Stream 所不具備的。那么,有沒有辦法憑空創(chuàng)建一個“真正的”文件流?
三.思路
要想創(chuàng)建出“真正的”文件流,至少有正反 2 種思路:
給普通流添上文件相關(guān)的 meta 信息
先拿到一個真正的文件流,再改掉其數(shù)據(jù)和 meta 信息
顯然,前者更靈活一些,并且實現(xiàn)上能夠做到完全不依賴文件
文件流的生產(chǎn)過程
沿著憑空創(chuàng)造的思路,探究 fs.createReadStream API 的 內(nèi)部實現(xiàn) 之后發(fā)現(xiàn),生產(chǎn)文件流的關(guān)鍵過程如下:
function ReadStream(path, options) { // 1.打開path指定的文件 if (typeof this.fd !== 'number') this.open(); } ReadStream.prototype.open = function() { fs.open(this.path, this.flags, this.mode, (er, fd) => { // 2.拿到文件描述符并持有 this.fd = fd; this.emit('open', fd); this.emit('ready'); // 3.開始流式讀取數(shù)據(jù) // read來自父類Readable,主要調(diào)用內(nèi)部方法_read // ref: https://github.com/nodejs/node/blob/v10.16.3/lib/_stream_readable.js#L390 this.read(); }); }; ReadStream.prototype._read = function(n) { // 4.從文件中讀取一個chunk fs.read(this.fd, pool, pool.used, toRead, this.pos, (er, bytesRead) => { let b = null; if (bytesRead > 0) { this.bytesRead += bytesRead; b = thisPool.slice(start, start + bytesRead); } // 5.(通過觸發(fā)data事件)吐出一個chunk,如果還有數(shù)據(jù),process.nextTick再次this.read,直至this.push(null)觸發(fā)'end'事件 // ref: https://github.com/nodejs/node/blob/v10.16.3/lib/_stream_readable.js#L207 this.push(b); }); };
P.S.其中第 5 步相對復(fù)雜, this.push(buffer) 既能觸發(fā)下一個 chunk 的讀?。?nbsp;this.read() ),也能在數(shù)據(jù)讀完之后(通過 this.push(null) )觸發(fā) 'end' 事件,具體見 node/lib/_stream_readable.js
重新實現(xiàn)文件流
既然已經(jīng)摸清了文件流的生產(chǎn)過程,下一步自然是 替換掉所有文件操作,直至文件流的實現(xiàn)完全不依賴文件 ,例如:
// 從文件中讀取一個chunk fs.read(this.fd, pool, pool.used, toRead, this.pos, (er, bytesRead) => { /* ... */ }); // 換成 this._fakeReadFile(this.fd, pool, pool.used, toRead, this.pos, (bytesRead) => { /* ... */ }); // 從輸入字符串對應(yīng)的Buffer中copy出一個chunk ReadStream.prototype._fakeReadFile = function(_, buffer, offset, length, position, cb) { position = position || this.input._position; // fake read file async setTimeout(() => { let bytesRead = 0; if (position < this.input.byteLength) { bytesRead = this.input.copy(buffer, offset, position, position + length - 1); this.input._position += bytesRead; } cb(bytesRead); }, 0); }
即從中剔除文件操作,用基于字符串的操作去替代它們
四.解決方案
如此這般,就有了 ayqy/string-to-file-stream ,用來憑空創(chuàng)建文件流:
string2fileStream('string-content') === fs.createReadStream(/* path to a text file with content 'string-content' */)`
例如:
const string2fileStream = require('string-to-file-stream'); const input = 'Oh, my great data!'; const s = string2fileStream(input); s.on('data', (chunk) => { assert.equal(chunk.toString(), input); }); 生成的流同樣能夠具有文件 meta 信息: const string2fileStream = require('string-to-file-stream'); const formData = new FormData(); formData.append('file', string2fileStream('my-string-data', { path: './abc.txt' })); form.submit('example.org/upload', function(err, res) { console.log(res.statusCode); });
看完上述內(nèi)容是否對您有幫助呢?如果還想對相關(guān)知識有進(jìn)一步的了解或閱讀更多相關(guān)文章,請關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,感謝您對創(chuàng)新互聯(lián)的支持。