浅谈Node.js:fs文件系统模块


Posted in Javascript onDecember 08, 2016

fs文件系统模块,这是一个非常重要的模块,对文件的操作都基于它。该模块的所有方法都有同步和异步两种方式,下面便介绍一下该模块的使用。

1、检测当前进程对文件的权限

使用fs.access(path[, mode], callback)方法检查权限,mode参数是一个整数,有以下常量值:

  • fs.constants.F_OK     path对调用进程是可见的,既存在
  • fs.constants.R_OK     path是可读的
  • fs.constants.W_OK    path是可写的
  • fs.constants.X_OK     path是可执行的

 使用如下所示:

fs.access('./note.txt',fs.constants.F_OK,(err)=>{
console.log(err?'文件不存在':'文件已经存在');
});

同步版本,如果发生异常,则直接抛出异常,否则什么也不做。同步版本可以利用try..catch来做,适用所有方法,如下所示:

try{
fs.accessSync('./note.txt',fs.constants.F_OK);
}catch(ex){
console.log('文件不存在');
}

2、获取文件状态

使用fs.stat(path, callback)fs.statSync(path)方法来获取指定path的状性,callback有(err, stats)两个参数,stats是fs.stats对象,具有以下属性:

{ dev: 638212,
 mode: 33206,
 nlink: 1,
 uid: 0,
 gid: 0,
 rdev: 0,
 blksize: undefined,
 ino: 105553116266564850,
 size: 1094,
 blocks: undefined,
 atime: 2016-11-22T08:45:43.505Z,
 mtime: 2016-11-22T09:33:13.535Z,
 ctime: 2016-11-22T09:33:13.535Z,
 birthtime: 2016-11-22T08:45:43.505Z }

还有以下方法:

stats.isFile()
 stats.isDirectory()
 stats.isBlockDevice()
 stats.isCharacterDevice()
 stats.isSymbolicLink() (only valid with fs.lstat())
 stats.isFIFO()
 stats.isSocket()

使用如下所示:

fs.stat('./app.js',(err,stats)=>{
  if(err) throw err;
  console.log(stats);
});
var stats = fs.statSync('../test.txt');//同步版本

3、文件追加

使用fs.appendFile(file, data[, options], callback)方法向file写入数据,如果file不存在,则创建file,data参数为字符串或buffer,options可选参数是对象或字符串,具有以下属性:

  • encoding  |  default = 'utf8' 编码
  • mode  default = 0o666 打开模式
  • flag  default = 'a'

使用如下所示:

fs.appendFile('./test.txt','hello world!\r\n',(err)=>{
  if(err) throw err;
  console.log('写入成功');
});
//appendFile同步版本,返回值为undefined
fs.appendFileSync('./test.txt','hello nodejs!\r\n');

4、文件读取和写入

文件读取使用fs.readFile(file[, options], callback)方法,参数含义如下:

  • file 文件名或文件描述符
  • options 对象或字符串,如果是对象,则包含encoding和flag,前者默认为null,后者为'r'
  • callback 参数为(err,data)

如果指定的文件不存在,则直接抛出错误。使用如下所示:

fs.readFile('./test4.txt',{encoding:'utf8',flag:'r'},(err,data)=>{
if(err) throw err;
console.log(data);
});
var data = fs.readFileSync('../test4.txt',{encoding:'utf8',flag:'r'});

文件写入数据,使用fs.writeFile(file, data[, options], callback)方法,参数含义如下:

  • file 文件名或文件描述符
  • data 字符串或buffer
  • options 对象或字符串,如果是对象,则包含encoding、mode以及flag,依次默认为utf8,0o666,'w'
  • callback 参数err

如果指定的文件不存在,则创建该文件,相反则替换原来的文件。使用如下所示:

var data = "hello node!";
fs.writeFile('./test1.txt',data,{flag:'w'},(err)=>{
if(err) throw err;
console.log('written ok.');
});
fs.writeFileSync('./test1.txt',data,{flag:'w'});

我们也可以利用fs的open,read,write,stat等方法来实现文件的读取和写入。

fs.open(path, flags[, mode], callback)方法打开一个文件获取句柄,flags参数有以下这些:

  • 'r' - 以只读方式打开文件,若文件不存在则报错。
  • 'r+' - 以读写方式打开文件,若文件不存在则报错。
  • 'rs+' 在同步模式下,以读写方式打开文件
  • 'w' - 以写方式打开文件,若文件不存在则创建
  • 'wx' - 以写方式打开文件,若文件不存在则抛出异常.
  • 'w+' - 以读写方式打开文件,若文件不存在则创建,相反则清空文件.
  • 'wx+' - 以读写方式打开文件,若文件不存在则抛出异常.
  • 'a' - 以追加方式打开文件,若文件不存则创建文件
  • 'ax' - 以追加方式打开文件,若文件不存则抛出异常.
  • 'a+' - 以追加和读方式打开文件,若文件不存则创建文件
  • 'ax+' - 以追加和读方式打开文件,若文件不存则失败

callback回调函数有(err,fd)两个参数。

fs.read(fd, buffer, offset, length, position, callback)方法,从一个文件中读取数据存入buffer中,参数含义如下:

  • buffer Buffer对象,用来存储读取的数据
  • offset buffer开始写的位置
  • length 需要读取的长度
  • position 指定从文件的哪个位置开始读取,若设置为null,则从文件当前位置开始读取
  • callback 有三个参数(err, bytesRead, buffer) bytesRead为实际读取字节数
  • fs.write(fd, buffer, offset, length[, position], callback)方法,将buffer数据写如指定文件中,参数含义如下:
  • offset和length指定buffer的部分
  • position 指定文件的开始写入的文件,若不为数字则从文件当前位置开始写入

 下面是一个使用open,write,read,stat方法实现的文件内容复制的函数,如下所示:

function copy(src, dest) {
const destFd = fs.openSync(dest, 'w+'),
   srcStat = fs.statSync(src);
const buffer = new Buffer(srcStat.size);
console.log('复制开始...');
console.log(src+'大小:'+srcStat.size)
fs.open(src,'r',(err,fd)=>{
  if(err) throw err;
  fs.read(fd,buffer,0,srcStat.size,null,(err,bytesRead,buff)=>{
    if(err) throw err;
    console.log('实际读取大小:'+bytesRead);
    fs.close(fd,()=>{});
    fs.write(destFd,buff,0,bytesRead,null,(err, written, buffer)=>{
      if(err) throw err;
      console.log('已完成复制,向'+dest+'写入了'+written);
      fs.close(destFd,()=>{});
    });
  });
});
}
copy('./app.js','./appbak.js');

执行结果如下:

E:\developmentdocument\nodejsdemo>node fs-examples.js
复制开始...
 ./app.js大小:1094
实际读取大小:1094
已完成复制,向./appbak.js写入了1094

5、文件重命名、删除

方法fs.rename(oldPath, newPath, callback)可以实现文件的重命名,还能实现文件的移动,如果oldPath与newPath在同一目录下,则是重命名,否则是移动文件并重命名,使用如下所示:

fs.rename('../test4.txt','./test4.txt',(err)=>{
  if(err) throw err;
  console.log('rename success.');
});
fs.renameSync('../test2.txt','../test4.txt');

文件删除需要用到fs.unlink(path, callback)方法,使用也很简单,如下所示:

fs.unlink('./dir/11.txt',(err)=>{
  if(err) throw err;
  console.log('delete file success.');
});
fs.unlinkSync('./dir/11.txt');

6、创建、读取、删除目录

创建目录使用的是fs.mkdir(path[, mode], callback)方法,mode参数默认为0o777,但是该方法只能创建一级目录,否则抛出异常,如下所示:

fs.mkdir('./a',0o777,(err)=>{
  if(err) throw err;
  console.log('mkdir success');
});
//mkdir的同步版本,返回值为undefined
fs.mkdirSync('./test',0o777);

为了能够创建多级目录,可以自己定义一个函数来实现,需要用到path模块的dirname方法,如下所示:

function isFileExists(filePath){
  var bool = !0;
  try{
    fs.accessSync(filePath,fs.F_OK);
  }catch(err){
    bool = !1;
  }
  return bool;
}
function mkdirp(dirpath,mode,cb){
  if(isFileExists(dirpath)){
    cb(dirpath);
  }else{
    mkdirp(path.dirname(dirpath),mode,function(){
      fs.mkdir(dirpath,mode,cb);
    });
  }
}

扫描目录需要用到fs.readdir(path[, options], callback)方法,options参数为字符串或对象,callback回调函数有(err, files)两个参数,files是一个文件名数组,该方法也是只能扫描一级目录,使用如下所示:

fs.readdir('./',(err,files)=>{
  if(err) throw err;
  console.log(files);
});

如果要实现可以递归扫描目录,可以自己定义一个函数,如下所示:

function scandir(dirpath){
  var filesArr = {};
  if(!isFileExists(dirpath)) return !1;
  function scan(filepath){
    var statObj = fs.statSync(filepath);
    if(!statObj.isDirectory()) return filesArr.push(filepath);
    var files = fs.readdirSync(filepath);
    files.forEach((file,idx,arr)=>{
      scan(filepath+'/'+file);
    });
  }
  scan(dirpath);
  return filesArr;
}

删除目录使用fs.rmdir(path, callback)方法,只能删除一级目录且目录须为空,使用如下所示:

fs.rmdir('./dir',(err)=>{
  if(err) throw err;
  console.log('delete dir success.');
});

要实现类似rm -rf的递归删除效果,可以使用如下代码:

function deldirs(dirpath){
  var stat = null,
    emptyFoldersArr = [];
  
  function scan(spath){
    var files = fs.readdirSync(spath);
    emptyFoldersArr.push(spath);
    if(files.length>0){
      files.forEach((file,idx,arr)=>{
        if(fs.statSync(spath+'/'+file).isDirectory()){
          scan(spath+'/'+file);
        }else{
          return fs.unlinkSync(spath+'/'+file),!0;
        }
      });
    }
  }
  scan(dirpath);
  for(var l=emptyFoldersArr.length-1,i=l;i>=0;i--){
    fs.rmdirSync(emptyFoldersArr[i]);
  }
}

7、获取路径的绝对路径

使用fs.realpath(path[, options], callback)方法可以获取path的绝对路径,callback有(err, resolvedPath)两个参数,使用如下所示:

fs.realpath('./test.txt',(err,resolvePath)=>{
  if(err) throw err;
  console.log(resolvePath);
});
console.log(fs.realpathSync('./test.txt'));

执行结果如下所示:

E:\developmentdocument\nodejsdemo>node fs-examples.js
E:\developmentdocument\nodejsdemo\test.txt

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
JavaScript 类型的包装对象(Typed Wrappers)
Oct 27 Javascript
浅析JavaScript声明变量
Dec 21 Javascript
总结jQuery插件开发中的一些要点
May 16 Javascript
Bootstrap导航条可点击和鼠标悬停显示下拉菜单
Nov 25 Javascript
JQuery Dialog对话框 不能通过Esc关闭的原因分析及解决办法
Jan 18 Javascript
ES6扩展运算符的用途实例详解
Aug 20 Javascript
JSON在Javascript中的使用(eval和JSON.parse的区别)详细解析
Sep 05 Javascript
Angularjs自定义指令实现分页插件(DEMO)
Sep 16 Javascript
分享vue.js devtools遇到一系列问题
Oct 24 Javascript
Vue 进阶之路(三)
Apr 18 Javascript
vue如何获取自定义元素属性参数值的方法
May 14 Javascript
基于element-ui对话框el-dialog初始化的校验问题解决
Sep 11 Javascript
Bootstrap基本插件学习笔记之模态对话框(16)
Dec 08 #Javascript
概述BootStrap中role="form"及role作用角色
Dec 08 #Javascript
Bootstrap基本插件学习笔记之标签切换(17)
Dec 08 #Javascript
Bootstrap基本插件学习笔记之Popover提示框(19)
Dec 08 #Javascript
JS焦点图,JS 多个页面放多个焦点图的实例
Dec 08 #Javascript
BootStrap 图标icon符号图标glyphicons不正常显示的快速解决办法
Dec 08 #Javascript
Bootstrap基本插件学习笔记之Tooltip提示工具(18)
Dec 08 #Javascript
You might like
Windows下的PHP5.0安装配制详解
2006/09/05 PHP
关于JSON以及JSON在PHP中的应用技巧
2013/11/27 PHP
为数据添加append,remove功能
2006/10/03 Javascript
js 鼠标拖动对象 可让任何div实现拖动效果
2009/11/09 Javascript
Javascript 面向对象 继承
2010/05/13 Javascript
JavaScript实现简单的时钟实例代码
2013/11/23 Javascript
实例讲解JS中数组Array的操作方法
2014/05/09 Javascript
Visual Studio中js调试的方法图解
2014/06/30 Javascript
js读取csv文件并使用json显示出来
2015/01/09 Javascript
jQuery()方法的第二个参数详解
2015/04/29 Javascript
javascript元素动态创建实现方法
2015/05/13 Javascript
基于Bootstrap实现图片轮播效果
2016/05/22 Javascript
node.js cookie-parser 中间件介绍
2016/06/06 Javascript
jQuery实现调整表格单列顺序完整实例
2016/06/20 Javascript
javascript中this关键字详解
2016/12/12 Javascript
javascript 面向对象function详解及实例代码
2017/02/28 Javascript
基于vue和react的spa进行按需加载的实现方法
2018/09/29 Javascript
vue-resource:jsonp请求百度搜索的接口示例
2019/11/09 Javascript
微信小程序如何实现radio单选框单击打勾和取消
2020/01/21 Javascript
vue使用vant中的checkbox实现全选功能
2020/11/17 Vue.js
Python入门_浅谈for循环、while循环
2017/05/16 Python
Python函数式编程
2017/07/20 Python
python中实现指定时间调用函数示例代码
2017/09/08 Python
Django项目中包含多个应用时对url的配置方法
2018/05/30 Python
不归路系列:Python入门之旅-一定要注意缩进!!!(推荐)
2019/04/16 Python
如何基于Python创建目录文件夹
2019/12/31 Python
Python原始套接字编程实例解析
2020/01/29 Python
TensorFlow:将ckpt文件固化成pb文件教程
2020/02/11 Python
关于赌博的检讨书
2014/01/24 职场文书
五年后的职业生涯规划
2014/03/04 职场文书
保护环境建议书100字
2014/05/13 职场文书
债务纠纷委托书
2014/08/30 职场文书
广告公司文案策划岗位职责
2015/04/14 职场文书
常住证明范本
2015/06/23 职场文书
Tomcat弱口令复现及利用
2022/05/06 Servers
Redis唯一ID生成器的实现
2022/07/07 Redis