常见拷贝操作
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'))