前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >【nodejs】I\u002FO,Buffer,http模块,stream

【nodejs】I\u002FO,Buffer,http模块,stream

作者头像
一尾流莺
发布2022-12-10 13:15:25
3840
发布2022-12-10 13:15:25
举报

I/O处理

关于 I/O ,有一个很经典的响水壶解释。

  • 隔壁王大爷有个水壶,王大爷经常用它来烧开水。
  • 同步阻塞:王大爷把水壶放到火上烧,然后啥也不干在那等,直到水开了王大爷再去搞别的事情。
  • 同步非阻塞:王大爷觉得自己有点憨,不打算等了。把水壶放上去之后大爷就是去看电视,时不时来瞅一眼有没有开。
  • 异步阻塞:王大爷去买了个响水壶,他把响水壶放在火上,然后也是等着水开,水开的时候水壶会发出声响。
  • 异步非阻塞:王大爷又觉得自己有点憨,他把响水壶放在火上然后去看电视,这时他不用是不是来瞅一眼,因为水开 的时候水壶会发出声音通知大爷。
  • 上面四个例子里,阻塞非阻塞说明的是大爷的状态,同步非同步说明的是水壶的调用姿势。水壶能在烧好的时候主动响起,就等同于我们异步的定义,能在结束时通知主线程并且回调。所以异步一般配合非阻塞,才能发挥其作用。

promisify

  • 使用 promisify 可以让一个回调函数风格的异步方法变成 promise 风格。
  • 利用一个自执行函数,就可以使用 async await 风格书写代码。
(async () => {
  const fs = require("fs")
  // 使用 promisify 可以让一个回调函数形式的异步方法 变成 promise 形式 从而可以使用 async await
  const { promisify } = require("util")
  // 将 fs.readFile 方法 包装成 promise
  const readFile = promisify(fs.readFile)
  // 读取的是个 buffer 对象 ,是 js 用来描述二进制的东西
  const data = await readFile("./test.js")
  // toString方法解析字符串
  console.log("data", data.toString())
})()

Buffer

  • 读取数据类型为 Buffer
  • 用于在 TCP 流、文件系统操作、以及其他上下文中与八位字节流进行交互。 八位字节组成的数组,可以有效的在 JS 中存储二进制数据。
// 创建一个长度为 10 字节以 0 填充的 Buffer
const buf1 = Buffer.alloc(10)
console.log('🚀🚀~ buf1:', buf1);

// 创建一个 Buffer 包含 ascii.
const buf2 = Buffer.from("a")
console.log('🚀🚀~ buf2:', buf2);

// 创建Buffer包含UTF-8字节 
// UFT-8:一种变长的编码方案,使用 1~6 个字节来存储; 
// UFT-32:一种固定长度的编码方案,不管字符编号大小,始终使用 4 个字节来存储; 
// UTF-16:介于 UTF-8 和 UTF-32 之间,使用 2 个或者 4 个字节来存储,长度既固定又可变。
const buf3 = Buffer.from('中')
console.log('🚀🚀~ buf3:', buf3);

// 写入 Buffer 数据 
buf1.write('hello'); 
console.log(buf1); 

// 读取 Buffer 数据 
console.log(buf3.toString());

// 合并 Buffer 数据 
const buf4 = Buffer.concat([buf2, buf3])
console.log('🚀🚀~ buf4:', buf4, buf4.toString());

http模块

  • writeHead 可以同时书写 statusCodesetHeader
  • 判断 req.headers.accept 是否包含 image/* 可以判断是不是请求图片。
const http = require("http")
const fs = require("fs")
// 创建服务
const server = http.createServer((req, res) => {
  const { url, method, headers } = req
  if (url === '/' && method === 'GET') {
    fs.readFile("./src/03-http/index.html", (err, data) => {
      if (err) {
        res.writeHead(500, { 'Content-Type': 'text/plain;charset=utf-8' })
        res.end("500 服务器错误")
        return
      }
      res.statusCode = 200
      res.setHeader('Content-Type', 'text/html')
      res.end(data)
    })
  }
  else if (url === '/getUser' && method === 'GET') {
    res.writeHead(200, { 'Content-Type': 'application/json' })
    res.end(JSON.stringify({ name: "warbler" }))
  }
  // Accept代表发送端(客户端)希望接受的数据类型。 比如:Accept:text/xml; 代表客户端希望 接受的数据类型是xml类型。 
  // Content-Type代表发送端(客户端|服务器)发送的实体数据的数据类型。 比如:ContentType:text/html; 代表发送端发送的数据格式是html。 
  // 二者合起来, Accept:text/xml; Content-Type:text/html ,即代表希望接受的数据类型是xml格 式,本次请求发送的数据的数据格式是html。
  else if (method === 'GET' && headers.accept.indexOf("image/*") !== -1) {
    // 统一描述所有的图片请求
    fs.createReadStream(`./src/03-http/${url}`).pipe(res)
  }
  else {
    res.statusCode = 404
    res.setHeader('Content-Type', 'text/plain;charset=utf-8')
    res.end("404 页面未找到")
  }
})
server.listen(3000, () => {
  console.log('🚀🚀~ sever at 3002');
})

stream

const fs = require("fs")

// 读流
const rs = fs.createReadStream('./src/04-stream/main.jpg')

// 写流
const ws = fs.createWriteStream('./src/04-stream/main2.jpg')

// 连接导管 可以复制图片到指定目录
rs.pipe(ws)

参考资料

开课吧全栈架构师课程

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2022-01-19,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • I/O处理
  • promisify
  • Buffer
  • http模块
  • stream
  • 参考资料
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档