web-dev-qa-db-ja.com

Node.jsの「終了後に書き込み」エラー

基本的なノードアプリから始めて、いくつかのサイトで提案されたコールバックがあっても、この「終了後に書き込む」エラーを克服する方法を理解できません。

index.js:

var server = require("./server");
var router = require("./router");
var requestHandlers = require("./requestHandlers");

var handle = {}
handle["/"] = requestHandlers.start;
handle["/start"] = requestHandlers.start;
handle["/upload"] = requestHandlers.upload;

server.start(router.route, handle);

server.js:

var http = require("http");
var url = require("url");

function start(route, handle) {
  function onRequest(request, response) {
    var postData ="";
    var pathname = url.parse(request.url).pathname;
    console.log("Request for " + pathname + " received.");

    route(handle, pathname, response);
   }

  http.createServer(onRequest).listen(8888);
  console.log("Server has started.");
}

exports.start = start;

router.js:

function route(handle, pathname, response) {
  console.log("About to route a request for " + pathname);
  if (typeof handle[pathname] === 'function') {
    handle[pathname](response);
  } 
  else {
    console.log("No request handler found for " + pathname);
    response.writeHead(404, {"Content-Type": "text/plain"});
    response.write("404 Not found");
    response.end();
  }
}

exports.route = route;

and now the culprit, requestHandlers.js:

function start(response) {
  console.log("Request handler 'start' was called.");

  var body = '<html>'+
    '<head>'+
    '<meta http-equiv="Content-Type" content="text/html; '+
    'charset=UTF-8" />'+
    '</head>'+
    '<body>'+
    '<div>Hello Start</div>' +
    '</body>'+
    '</html>';

    response.writeHead(200, {"Content-Type": "text/html"});
    response.write(body);
    response.end();
}

function upload(response) {
    console.log("Request handler 'upload' was called");
    response.writeHead(200, {"Content-Type": "text/plain"}); 
    response.write("Hello Upload");
    response.end();
}

exports.start = start;
exports.upload = upload;

そして、これがスタックトレースです:

C02RF8L7FVH5:HelloWorld francop $ nodeindex.jsサーバーが起動しました。リクエスト/受信。 /リクエストハンドラ「start」のリクエストをルーティングしようとしています。 /リクエストハンドラ「start」のリクエストをルーティングしようとしています。 events.js:141 throw er; //未処理の「エラー」イベント^

エラー:終了後にServerResponse.OutgoingMessage.write(_http_outgoing.js:430:15)のObject.start(/Users/francop/Documents/Apps/HelloWorld/requestHandlers.js:18:14)のルート(/ Users/francop/Documents/Apps/HelloWorld/router.js:5:21)IncomingMessageで。 (/Users/francop/Documents/Apps/HelloWorld/server.js:19:6)EmitNone(events.js:67:13)at IncomingMessage.emit(events.js:166:7)at endReadableNT(_stream_読み取り可能.js :921:12)at nextTickCallbackWith2Args(node.js:442:9)at process._tickCallback(node.js:356:17)

私がどこかで見つけた唯一の答えは、次のように「非同期なので、コールバックを使用する」でした。

response.write(body, function(err) {
    response.end();
});

...まったく同じエラーをスローします。

これに関するどんな助けも素晴らしいでしょう。私がオンラインで見つけたいくつかのまばらな答えはまったく役に立ちません。

ありがとう!

7
sir_dancealot

response.end()ではなくresponse.write()で体を返します

function start (response) {
    console.log("Request handler 'start' was called.");
    var body = '<html>'+
    '<head>'+
    '<meta http-equiv="Content-Type" content="text/html; '+
    'charset=UTF-8" />'+
    '</head>'+
    '<body>'+
    '<div>Hello Start</div>' +
    '</body>'+
    '</html>';

    response.writeHead(200, {"Content-Type": "text/html"});
    response.end(body);
}

編集1:

response.write()が初めて呼び出されると、バッファリングされたヘッダー情報と最初の本文がクライアントに送信されます。 2回目にresponse.write()が呼び出されると、Node.jsはデータをストリーミングすることを想定し、それを個別に送信します。つまり、応答は本文の最初のチャンクまでバッファリングされます。

したがって、1つのチャンクのみを送信する場合は、response.end(body)を使用する必要があり、複数のチャンクを送信する場合は、複数のresponse.end()の後にresponse.write()で呼び出しを終了する必要があります。 。

単一チャンクの例:

function (chunk){
    response.end(chunk);
}

複数のチャンクの例:

function (chunkArray){
    // loop on chunks 
    for(var i=0; i < chunkArray.length; i++){
        response.write(chunkArray[i]);                
    }

    // end the response with empty params
    response.end();
}
4
David Antoon

これは「The Node Beginner」の例です。

この投稿の時点では、「router.js」と「requestHanders.js」の間に分離がないため、以下のファイルを再投稿しました。

これをノードv8で実行したときに同じ問題が発生していましたが、現在、インシデントノードv9.5.0なしで実行されています。私はそれが私のために働いたと言っているだけで問題の解決策としてこれを提示していません。おそらくこれは、更新されたバージョンで解決された問題であり、はっきりとは言えません。

また、エラーストリームには「requestHandlers.js:18:14」と表示されます。コードの18行目を確認すると、より良い結果が得られる可能性があります(将来)。スペースなしの18行目は、アップロード関数の宣言前の行です。実際、このエラーは「response.end();」によってトリガーされました。

index.js

var server = require("./server");
var router = require("./router");
var requestHandlers = require("./requestHandlers");

var handle = {}
handle["/"] = requestHandlers.start;
handle["/start"] = requestHandlers.start;
handle["/upload"] = requestHandlers.upload;

server.start(router.route, handle);

router.js

function route(handle, pathname, response) {
  console.log("About to route a request for " + pathname);
  if (typeof handle[pathname] === 'function') {
    handle[pathname](response);
  } 
  else {
    console.log("No request handler found for " + pathname);
    response.writeHead(404, {"Content-Type": "text/plain"});
    response.write("404 Not found");
    response.end();
  }
}

exports.route = route;

server.js

var http = require("http");
var url = require("url");

function start(route, handle) {
  function onRequest(request, response) {
    var postData ="";
    var pathname = url.parse(request.url).pathname;
    console.log("Request for " + pathname + " received.");

    route(handle, pathname, response);
   }

  http.createServer(onRequest).listen(8888);
  console.log("Server has started.");
}

exports.start = start;

requestHandlers.js

function start(response) {
  console.log("Request handler 'start' was called.");

  var body = '<html>'+
    '<head>'+
    '<meta http-equiv="Content-Type" content="text/html; '+
    'charset=UTF-8" />'+
    '</head>'+
    '<body>'+
    '<div>Hello Start</div>' +
    '</body>'+
    '</html>';

    response.writeHead(200, {"Content-Type": "text/html"});
    response.write(body);
    response.end();
}

function upload(response) {
    console.log("Request handler 'upload' was called");
    response.writeHead(200, {"Content-Type": "text/plain"}); 
    response.write("Hello Upload");
    response.end();
}

exports.start = start;
exports.upload = upload;
1
Nelles