Node.js で fs.readFile を fs.createReadStream に置き換える 質問する

Node.js で fs.readFile を fs.createReadStream に置き換える 質問する

ディレクトリから画像を読み取って index.html に送信するコードがあります。

fs.readFile を fs.createReadStream に置き換えようとしていますが、良い例が見つからないため、これをどのように実装すればよいかわかりません。

これが私が得たものです (index.js)

var app = require('express')();
var http = require('http').Server(app);
var io = require('socket.io')(http);

var fs = require('fs');

http.listen(3000, function () {
     console.log('listening on *:3000');
});
app.get('/', function (req, res) {
     res.sendFile(__dirname + '/public/views/index.html');
});
io.on('connection', function (socket) {
     fs.readFile(__dirname + '/public/images/image.png', function (err, buf){
        socket.emit('image', { image: true, buffer: buf.toString('base64') });
     });
});

インデックス.html

<!DOCTYPE html>
<html>
<body>

<canvas id="canvas" width="200" height="100">
    Your browser does not support the HTML5 canvas tag.
</canvas>

<script src="https://cdn.socket.io/socket.io-1.2.0.js"></script>

<script>
    var socket = io();
    var ctx = document.getElementById('canvas').getContext('2d');
    socket.on("image", function (info) {
        if (info.image) {
            var img = new Image();
            img.src = 'data:image/jpeg;base64,' + info.buffer;
            ctx.drawImage(img, 0, 0);
        }
    });
</script>
</body >
</html >

ベストアンサー1

stream.Readable以下のアプローチでは、コア モジュールのみを使用し、から返されたインスタンスからチャンクを読み取りfs.createReadStream()、それらのチャンクを として返しますBuffer。チャンクをストリームで返さないのであれば、これはあまり良いアプローチではありません。メモリ内にある 内にファイルを保持することになるのでBuffer、適度なサイズのファイルにのみ適したソリューションです。

io.on('connection', function (socket) {
  fileToBuffer(__dirname + '/public/images/image.png', (err, imageBuffer) => {
    if (err) { 
      socket.emit('error', err)
    } else {
      socket.emit('image', { image: true, buffer: imageBuffer.toString('base64') }); 
    }
  });
});

const fileToBuffer = (filename, cb) => {
    let readStream = fs.createReadStream(filename);
    let chunks = [];

    // Handle any errors while reading
    readStream.on('error', err => {
        // handle error

        // File could not be read
        return cb(err);
    });

    // Listen for data
    readStream.on('data', chunk => {
        chunks.push(chunk);
    });

    // File is done being read
    readStream.on('close', () => {
        // Create a buffer of the image from the stream
        return cb(null, Buffer.concat(chunks));
    });
}

HTTP レスポンス ストリームの例

ほとんどの場合、ストリーミング データに使用する方がよい方法です。これはプロトコルに組み込まれているためで、ファイル ストリームを応答に直接送信HTTPできるため、データを一度にメモリに読み込む必要はありません。pipe()

これは、余計な機能のない非常に基本的な例であり、 を実行する方法を示すだけですpipe()stream.Readable例では Express を使用していますが、Node.js Core API からまたは をhttp.ServerResponse使用してもまったく同じように動作します。httphttps

const express = require('express');
const fs = require('fs');
const server = express();

const port = process.env.PORT || 1337;

server.get ('/image', (req, res) => {
    let readStream = fs.createReadStream(__dirname + '/public/images/image.png')

    // When the stream is done being read, end the response
    readStream.on('close', () => {
        res.end()
    })

    // Stream chunks to response
    readStream.pipe(res)
});

server.listen(port, () => {
    console.log(`Listening on ${port}`);
});

おすすめ記事