使用流时knex.js上的内存问题

Dur*_*den 6 sqlite stream out-of-memory node.js knex.js

我正在尝试使用knex.js将整个sqlite3数据库表导出为CSV .由于表可以达到300000行,我使用流来解决内存问题.但如果我查看我的应用程序的内存使用量高达800MB或我有一个"内存不足"错误.

如何在sqlite3数据库上使用knex.js处理大型查询结果?

下面的代码示例:

knex.select().from(table).stream(function (stream) {
    var stringifier = stringify(opts);
    var fileStream = fs.createWriteStream(file);

    var i = 0;
    stringifier.on('readable', function() {
      var row;
      while (row = stringifier.read()) {
        fileStream.write(row);
        console.log("row " + i++); //debug
      }
    });

    fileStream.once('open', function(fd) {
      stream.pipe(stringifier);
    });
});
Run Code Online (Sandbox Code Playgroud)

编辑

似乎sqlite3数据库的knex.js流是"假"流.在knex中sqlite3的流函数的源代码下面:

Runner_SQLite3.prototype._stream = Promise.method(function(sql, stream, options) {
    /*jshint unused: false*/
    var runner = this;
    return new Promise(function(resolver, rejecter) {
        stream.on('error', rejecter);
        stream.on('end', resolver);
        return runner.query(sql).map(function(row) {
            stream.write(row);
        }).catch(function(err) {
            stream.emit('error', err);
        }).then(function() {
            stream.end();
        });
    });
});
Run Code Online (Sandbox Code Playgroud)

我们看到它在从结果数组创建流之前等待执行请求.

版:

  • Knex.Js 0.7.5
  • 节点0.12

谢谢你的帮助.

Dur*_*den 5

我认为没有解决方案.我使用limit和offset来逐步使用knex.js获取所有数据,并在写入流中写入每一行.想要的人的实现示例:

 exportTable: function(table, writeStream) {
    var totalRows;
    var rowLimit = _config.ROW_LIMIT;

    return DatabaseManager.countAll(table).then(function(count) {

        totalRows = count[0]['count(*)'];
        var iterations = new Array(Math.ceil(totalRows / rowLimit));

        return Promise.reduce(iterations, function(total, item, index) {

            return _knex.select().from(table).limit(rowLimit).offset(index * rowLimit).map(function(row) {
                writeStream.write(row);
            }).catch(function(err) {
                return Promise.reject(err);
            });

        }, 0).then(function() {
            return Promise.resolve();
        }).catch(function(err) {
            return Promise.reject(err);
        });

    }).catch(function(err) {
        console.log(err);
        return Promise.reject(err);
    });
}
Run Code Online (Sandbox Code Playgroud)