首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >用csv写流“错误:结束后写”

用csv写流“错误:结束后写”
EN

Stack Overflow用户
提问于 2018-07-14 02:38:56
回答 2查看 4.4K关注 0票数 1

我确信这是对溪流工作方式的根本误解,但我的头却撞到了墙上。

我有一些json格式的传感器数据,我想使用csv -写流包附加到csv文件中。数据作为post请求发送到节点服务器,目的是将其附加到csv文件中。它第一次将一行写到csv文件中很好,但是如果我尝试发送另一个post请求,则会得到“错误:结束后写”错误。

代码语言:javascript
复制
function write_csv(obj) {
    writer.pipe(fs.createWriteStream('tides.csv', { flags: 'a' }));
    writer.write(obj);
    writer.end();
};

如果我注释掉"writer.end()“,它可以正常工作,但这不会最终抛出一个内存错误吗?如果是,那么添加到csv文件并避免此错误的正确方法是什么?

编辑:这是整个server.js文件

代码语言:javascript
复制
const express = require('express');
const bodyParser = require('body-parser');
const path = require('path');
const exphbs = require('express-handlebars');
const fs = require('fs');
const csvWriter = require('csv-write-stream');

const writer = csvWriter({ sendHeaders: false });
const app = express();

app.set('views', path.join(__dirname, 'views'));
app.engine('handlebars', exphbs({ defaultLayout: 'main' }));
app.set('view engine', 'handlebars');

app.set('port', (process.env.PORT || 3000));

app.use(express.static(path.join(__dirname, 'public')));

app.use(bodyParser.urlencoded({ extended: false }))
app.use(bodyParser.json())

app.get('/', function (req, res) {
    res.render('home')
})

app.post('/test', function (req, res, next) {
    // console.log("post received");
    distance = req.body.distance;
    let result = test(distance);
    let result_str = JSON.stringify(result);
    res.end(result_str)
});

function write_csv(obj) {
    writer.pipe(fs.createWriteStream('out.csv', { flags: 'a' }));
    writer.write(obj);
    writer.end();
};

function test(dist) {
    let d = new Date();
    let YYYY = d.getFullYear();
    let MM = d.getMonth();
    let DD = d.getDate();
    let HH = d.getHours();
    let mm = d.getMinutes();
    let ss = d.getSeconds();
    let date = YYYY + ':' + (MM + 1) + ':' + DD + ':' + HH + ':' + mm + ':' + ss;
    let time_distance = { 'time': date, 'distance': distance };
    console.log(time_distance);
    write_csv(time_distance);
    return time_distance;
};


app.listen(app.get('port'), function () {
    console.log('Sever started on port ' + app.get('port'));
})
EN

回答 2

Stack Overflow用户

回答已采纳

发布于 2018-07-14 03:22:45

没有看到完整的代码,我可以想象您正在多次调用write_csv,因为您试图将多个对象写入该文件。

问题是,当您第一次调用write_csv时,您将结束writer,这就是为什么您第二次调用它的原因是:

代码语言:javascript
复制
Error [ERR_STREAM_WRITE_AFTER_END]: write after end
代码语言:javascript
复制
function write_csv(obj) {
    writer.pipe(fs.createWriteStream('out.csv', { flags: 'a' }))
    writer.write()
    writer.end();
}
write_csv({ hello: 'world', foo: 'bar', baz: 'taco'});
// When you call it again, writer.end(); is already closed
// The following line will trigger the error
write_csv({ hello: 'world', foo: 'bar', baz: 'taco'});

相反,你应该做的是,只有在你写完文章后,才能接近作者。

代码语言:javascript
复制
const writer = csvWriter(); // Create new writer
// open file
writer.pipe(fs.createWriteStream('out.csv', { flags: 'a' }));

for(const obj of objects) // Write as many times as you wish
   writer.write(obj);

writer.end(); // I'm done writing.

现在您遇到的问题是,如果您尝试执行多个.writes,您将达到内存限制,因为您没有正确地处理背压

我建议阅读以下问题:

为什么试图编写大文件会导致js堆耗尽内存?

要处理这个问题,您需要等待发出drain事件。

下面是csvWriter的包装器,它将处理背压。

代码语言:javascript
复制
const fs = require('fs');
const csvWriter = require('csv-write-stream');

class Writer {

    constructor(file) {
        this.writer = csvWriter();
        this.writer.pipe(fs.createWriteStream(file, { flags: 'a' }));
    }

    write(obj) {
        // if .write returns false we have to wait until `drain` is emitted
        if(!this.writer.write(obj))
            return new Promise(resolve => this.writer.once('drain', resolve))

        return true;
    }

    end() {
        // Wrap it in a promise if you wish to wait for the callback.
        this.writer.end(); 
    }

}

(async() => {
    const writer = new Writer('out.csv');

    for(let i = 0; i < 1e8; i++) {
        const res = writer.write({ hello: 'world', foo: 'bar', baz: 'taco' });
        if(res instanceof Promise) {
            // You can remove this if, and leave just: await writer.write...
            // but the code will be slower
            await res; // This will wait for the stream to emit the drain event
        }
    }

    writer.end();

})();

Update:现在使用实际的代码,上面的答案仍然有效,但因为您在收到请求时正在写入文件。您可以选择是否打开文件一次,并对每个请求进行写入,在服务器关闭时(或您选择时)关闭它。或者只是打开文件,写它,然后在每次请求时关闭它,

对于前者,您应该使用上面的答案,对于后者,您所需要做的就是每次调用write_csv时创建一个新的编写器,而不是让一个全局编写器。

代码语言:javascript
复制
function write_csv(obj) {
    // Create a new writer every time
    const writer = csvWriter({ sendHeaders: false }); 
    writer.pipe(fs.createWriteStream('out.csv', { flags: 'a' }));
    writer.write(obj);
    writer.end();
};
票数 2
EN

Stack Overflow用户

发布于 2018-07-14 03:59:43

你可以试试这段代码,我认为它解决了你的问题。在路由中使用此函数并传递json数据。

首先安装此软件包:npm i --save json2csv

代码语言:javascript
复制
const Json2csvParser = require("json2csv").Parser;
const fs = require("fs");

function csvConverter(jsonData, cb) {

  const json2csvParser = new Json2csvParser();
  const csv_data = json2csvParser.parse(jsonData);
  const file_name = "report";

  // TODO: Change file path accordingly
  const file_path = `/Users/public/csv_files/${file_name}.csv`;

  fs.appendFile(file_path, csv_data, err => {
    if (err) {
      console.log(err);
      cb(err, null);
      return;
    }
    const response = {
    msg: "successful",
    file_address: file_path
  };
    cb(null, response);
    return;
  });
}

票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/51334780

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档