我正在http://nodejs.org/docs/v0.4.0/api/http.html#http.request上阅读文档,但由于某种原因,我似乎无法在返回的已完成响应对象上找到body/data属性。

> var res = http.get({host:'www.somesite.com', path:'/'})

> res.finished
true

> res._hasBody
true

它完成了(http。Get为您完成这一工作),因此它应该有某种内容。但没有尸体,没有数据,我无法从中读取信息。尸体藏在哪里?


当前回答

如果你想用。get,你可以这样做

http.get(url, function(res){
    res.setEncoding('utf8');
    res.on('data', function(chunk){
        console.log(chunk);
    });

});

其他回答

编辑:6年后回复自己

await关键字是从HTTP请求获得响应的最佳方式,避免了回调和.then()

您还需要使用返回promise的HTTP客户机。get()仍然返回一个Request对象,因此这将不起作用。

fetch是一个低级客户端,从npm和nodeJS 17起都可以使用。 superagent是一个成熟的HTTP客户端,具有更合理的默认值,包括更简单的查询字符串编码,正确使用mime类型,默认为JSON,以及其他常见的HTTP客户端特性。 Axios也很受欢迎,具有与超级代理类似的优势

await将等待Promise有一个值—在本例中是一个HTTP响应!

const superagent = require('superagent');

(async function(){
  const response = await superagent.get('https://www.google.com')
  console.log(response.text)
})();

使用await,一旦superagent.get()返回的promise有值,控制就会简单地传递到下一行。

如果你想用。get,你可以这样做

http.get(url, function(res){
    res.setEncoding('utf8');
    res.on('data', function(chunk){
        console.log(chunk);
    });

});

主体并不是作为响应的一部分完全存储的,这是因为主体可以包含非常大量的数据,如果它被存储在响应对象上,程序的内存将很快被消耗。

相反,Node.js使用一种叫做流的机制。这种机制允许只保留一小部分数据,并允许程序员决定是在内存中完全使用它,还是将数据的每个部分作为其流使用。

有多种方法可以将数据完全消耗到内存中,因为HTTP Response是可读流,所以res对象上所有可读方法都可用

监听“data”事件并保存传递给回调的数据块 Const chunks = [] Res.on ("data", (chunk) => { chunks.push(块) }); Res.on ("end", () => { const body = Buffer.concat(chunk); });

当使用这种方法时,您不会干扰流的行为,而只是收集应用程序可用的数据。

使用"readble"事件并调用res.read() Const chunks = []; Res.on ("readable", () => { 让块; While (null !== (chunk = res.read())){ chunks.push(块) } }); Res.on ("end", () => { const body = Buffer.concat(chunk); });

当采用这种方法时,你完全负责流的流,直到res.read被调用,才会有更多的数据被传递到流中。

使用异步迭代器 Const chunks = []; For await (const chunk of readable) { chunks.push(块); } const body = Buffer.concat(chunk);

这种方法类似于“数据”事件方法。它将简化范围,并允许整个过程发生在相同的范围内。

虽然如上所述,完全使用响应中的数据是可能的,但如果确实有必要这样做,始终要记住这一点。在许多情况下,可以简单地将数据定向到目的地,而不将其完全保存到内存中。

Node.js读取流,包括HTTP响应,有一个内置的方法来做这件事,这个方法叫做pipe。用法很简单,readStream.pipe(writeStream);。

例如:

如果数据的最终目的地是文件系统,则可以简单地打开到文件系统的写流,然后将数据通过管道传输到其目的地。

const { createWriteStream } = require("fs");
const writeStream = createWriteStream("someFile");
res.pipe(writeStream);

你需要为请求添加一个监听器,因为node.js是异步工作的:

request.on('response', function (response) {
  response.on('data', function (chunk) {
    console.log('BODY: ' + chunk);
 });
});

只是一个改进版的nkron响应。

const httpGet = url => {
  return new Promise((resolve, reject) => {
    http.get(url, res => {
      res.setEncoding('utf8');
      const body = [];
      res.on('data', chunk => body.push(chunk));
      res.on('end', () => resolve(body.join('')));
    }).on('error', reject);
  });
};

在字符串[]中添加块对内存使用更好,join(")将分配 新的内存只有一次。