node.js 深入 5 fs对文件的操作,创建删除

首先要学习二叉搜索树的实现,可以看这篇文章二叉搜索树

fs对目录的操作

需掌握的api:

  • fs.mkdir, 异步创建文件夹,如果存在报错
  • fs.mkdirSync, 同步创建目录 如果存在报错
  • fs.existSync, 判断文件/文件夹是否存在
  • fs.stat 判断文件/文件夹的状态,不存在报错
  • fs.rmdir 异步删除文件夹,文件夹下有儿子报错
  • fs.rmdirSync 同步删除文件夹
  • fs.readdir 读取文件夹下的儿子,读取文件报错
  • fs.unlink 删除文件
  • fs.rename 重命名文件

创建目录

在这里插入图片描述
只能一层一层创建,而且当D存在的时候会报错。我们实现自己的Mkfir方法,使其可以跨层创建并且文件已存在也不会报错。
思路:通过递归,先一层一层创建出来,然后判断如果存在就不创建,继续递归。
在这里插入图片描述
实现自己的Mkdir方法,递归执行创建,通过fs.stat来判断是否创建还是跳过。在这里插入图片描述在这里插入图片描述

创建成功。
不过这种写法不够雅观,是通过回调函数的,可以通过promise来改写。
在这里插入图片描述
node11之后,不用通过util.promisfy来改写fs的方法了。在这里插入图片描述
这次不用递归了,同步最好使用循环,使用async使mkdir函数放回一个promise,然后await fs.mkdir同步执行,等待完成后返回success。如果mkdir中报错了,会被catch捕获到。
在这里插入图片描述
使用async+await的方式比回调简洁很多。

删除目录/文件夹/文件

fs.rmidr

删除文件夹,异步判断删除文件夹 但是只要文件夹不为空,就不能删掉, 而且删除的文件/文件夹不存在也会报错
在这里插入图片描述
在这里插入图片描述
fs.readdir 异步读取目录/文件夹中的儿子,只能读儿子,不存在会报错,不能读取文件在这里插入图片描述
在这里插入图片描述
fs.unlink(fs.rname)删除文件,是文件,不是文件夹。
现在有个需求,判断删除a。那么我们需要先判断a是文件还是文件夹,然后再调用对应的方法,否则会报错
在这里插入图片描述
通过stat判断文件状态,提供isFile和isDirectory来判断是否文件或者文件夹,调用fs.unlink或者fs.rmdir。

封装删除方法

封装删除方法,即使删除文件夹并且文件夹下面有儿子,也直接删除。
思路就是:目录的结构跟树结构很像,必须删除所有的儿子才能删除文件夹。通过递归遍历删除所有的儿子,在这里插入图片描述
如图就是我们应该做的删除顺序。
在这里插入图片描述
如图,思路可能有点绕。step的递归是通过回调来执行的。而rmdir本身也会递归。具体思路就是:

  • 1 先通过fs.stat判断删除文件还是文件夹,如果使文件,就直接删除,调用回调,这个调用回调很关键。
  • 2 如果是文件夹,就通过readdir获取文件夹下的所有目录,拼接起来,然后搞了索引,指向指向当前要被删除的儿子。
  • 3 接着编写一个step函数用来判断索引是否等于儿子数组的长度,来判断儿子是否清空,不是的话执行rmdir删除第一个儿子,因为不知道儿子是文件还是文件夹所以继续递归调用rmdir,并让索引++。
  • 4 step函数必须作为rmdir的回调,因为step是用来清空儿子的,只有通过回调调用step,让其删除完所有儿子,才会删除当前的文件夹,并且调用cb退出。
  • 这里的难点主要是rmdir的递归以及step的回调调用,rmdir用来删除文件/文件夹,而step用来删除文件夹下的所有儿子。
  • 这种方式叫做串行,效率较低。
并行删除

在这里插入图片描述
两棵树同时开始删。
在这里插入图片描述
这里i从递归调用rmidr和step变成了循环递归调用rmdir。
主要思路就是dirs同时并发删除,删除成功后触发回调,index++,当index等于当前长度之后,就表示儿子删除完毕,直接删除文件夹就好。
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
删除成功,这是并发与串行的区别。

转换成async await

换成同步代码
在这里插入图片描述

在这里插入图片描述
关键就是promise.all这个,他会并发执行dirs里面每个promise,里面的值又去调用syncRmdir,反复调用,直到最底层的文件夹被被删除,然后返回success,以此类推,到最后最外层的promise返回success,继续往下走执行await fsp.rmdir删除文件夹。因为async函数返回的就是一个Promise了。所以promise.all里面的每个值都是promise。
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
删除成 功。

总结:

可以自己通过回调函数的形式,串行或者并发删除子文件,最后再成功删除文件夹。也可以通过promise,async+await的形式通过promise.all,递归调用,同步删除所有子文件之后再去删除文件夹。

阶段性总结

流是用来解决大文件读取问题的,不像readFile,流可以指定读取的位置和读取的大小,客户端发送给服务端的数据都是一段段,接受客户端的数据,也是用流的方式。

流的模式

可写流:rs.on('data) rs.on(‘end’) push(数据)发送数据触发data事件
可写流: ws.wirte() ws.end() 只能sting|buffer ,指定写入内容的位置,将内容分开写入
双工流:tcp,基于net模块包装的tcp双工流,能读能写。socket.write socket.end socket.on(‘data’)
转化流: transform模式

pipe管道

pipe: 可以将读取到的内容,发送给写入,如rs.pipe(ws)
管道实现的思想:靠的是发布订阅模式,读取一段内容之后就发送给写入流。写入使多个异步并发,可以实现异步顺序造队列,实现按顺序依次执行,保证每次只有一个fs.write在执行。

文件操作

目录的结构跟树结构很相似。
fs.mkdir
fs.rmdir
fs.stat stateObj.isDirectory() stateObj.isFile()
fs.readdir
fs.unlink
fs.existSync

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

coderlin_

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值