fs模块

作者: 吴晗君 | 来源:发表于2019-04-27 11:43 被阅读0次

    常见拷贝操作

    
    function copy(sourceSrc, targetSrc) {
      fs.readFile(sourceSrc, (e, data) => {
        // 默认读取类型为buffer,可以指定encoding为utf-8,但是在copy里面一般不改变编码类型,如果原资源是图片呢
        if (e) return console.log(e)
        fs.writeFile(targetSrc, data, e => {
          // 这里一样。
          if (e) console.log(e)
        })
      })
    }
    
    copy(
      path.resolve(__dirname, './test/1.md'),
      path.resolve(__dirname, './test/2.md')
    )
    

    这样有个问题,如果文件过大,就会占用极大内存。

    fs.open、fs.read的用法

    let buffer = Buffer.alloc(9)
    // path 路径
    // flags 对文件的操作类型 w r + a
    // mode 权限 读取(4)、写入(2)、执行(1)
    // 默认为八进制0o666 也就是十进制的438
    // 444 分别对应自己、群组、其他
    fs.open(path.resolve(__dirname, './test/1.md'), 'r', 0o444, (err, fd) => {
      if (err) return console.log(err)
      // fd(file descriptor) 文件描述符,从fd指定的文件中读取数据(其实就是个文件id)
      // buffer 是数据将写入的缓冲区
      // offset 0 buffer中写入的偏移量
      // length 9 读取的字节数
      // position 0 指定从文件读取的位置
      fs.read(fd, buffer, 0, 9, 0, (err, bytesRead) => {
        if (err) return console.log(err)
        // bytesRead实际读取到的个数
        console.log(bytesRead)
        console.log(buffer.toString())
      })
    })
    

    fs.write的用法

    let buffer2 = Buffer.from('我是吴晗君')
    // r+表示读取并写入,如果没有文件会报错
    // w+表示读取并写入,如果没有文件会创建新文件,不会报错
    fs.open(path.resolve(__dirname, './test/3.md'), 'w+', (err, fd) => {
      if (err) return console.log(err)
      fs.write(fd, buffer2, 0, 15, (err, bytesWritten, buffer) => {
        if (err) return console.log(err)
        console.log('被写入的字节数:', bytesWritten)
        console.log('buffer:', buffer)
        console.log('bufferStr:', buffer.toString())
      })
    })
    

    拷贝一次

    // 不把open放在里面的原因是要递归读取,如果放在里面的话每次递归都要读取一遍。
    function copy2 (sourceSrc, targetSrc) {
      const memory = 3 // 每次都以3个字节的数量读取并写入
      const buffer = Buffer.alloc(memory)
      let offset = 0
      let position = 0
      fs.open(sourceSrc, 'r', (err, fd) => {
        if (err) return console.log(err)
        fs.read(fd, buffer, offset, memory, position, (err, bytesRead, buffer) => {
          if (err) return console.log(err)
          fs.open(targetSrc, 'w', (err, fd) => {
            if (err) return console.log(err)
            fs.write(fd, buffer, offset, memory, position, (err, bytesWritten, buffer) => {
              if (err) return console.log(err)
              console.log('被写入的字节数:', bytesWritten)
              console.log('buffer:', buffer)
              console.log('bufferStr:', buffer.toString())
              position += memory
            })
          })
        })
      })
    }
    
    

    递归拷贝文件流,减少内存占用

    // 拷贝 文件流
    function copy3 (sourceSrc, targetSrc) {
      const memoryLen = 3 // 每次都以3个字节的数量读取并写入
      const buffer = Buffer.alloc(memoryLen)
      let position = 0
    
      fs.open(sourceSrc, 'r', (err, readFd) => {
        if (err) return console.log(err)
        fs.open(targetSrc, 'w', (err, writeFd) => {
          if (err) return console.log(err)
          function next () {
            fs.read(readFd, buffer, 0, memoryLen, position, (err, bytesRead, buffer) => {
              if (err) return console.log(err)
              // 如果能读取到文件内容就进行复制,而不是想着得到整个文件内容的长度进而进行比较
              if (bytesRead > 0) { 
                // 这里用bytesRead而不是memoryLen的原因是如果最后一次读取长度不够3个字节,只有一个字节
                // 那么buffer内被写入的只能是那一个字节,后面两个字节不变。所以会多出一段内容
                fs.write(writeFd, buffer, 0, bytesRead, position, (err, bytesWritten, buffer) => {
                  if (err) return console.log(err)
                  console.log('被写入的字节数:', bytesWritten)
                  console.log('buffer:', buffer)
                  console.log('bufferStr:', buffer.toString())
                  position += memoryLen
                  next()
                })
              } else {
                // 读取完毕
                fs.close(readFd, () => {})
                fs.close(writeFd, () => {})
              }
            })
          }
          next()
        })
      })
    }
    copy3(path.resolve(__dirname, './test/1.md'), path.resolve(__dirname, './test/3.md'))
    

    相关文章

      网友评论

          本文标题:fs模块

          本文链接:https://www.haomeiwen.com/subject/ctnhnqtx.html