一、文件操作
1.1 读取文件
读取文本文件时,如.txt,.js,.json等文件,直接使用readFile就可以获取文件的内容。
if(err) throw err;
console.log(data);
});
读取图片时,我们是不能直接输出到控制台中的,是需要创建一个服务器,然后在浏览器上进行查看。其实在上节中,我们已经了解过显示图片的过程了。
// 使用二进制方式读取图片
fs.readFile('./img/test.png','binary',file){
if( err ) throw err;
// 当前数据以image/png方式进行输出
response.writeHead(200,{"Content-Type": "image/png"});
response.write(file,'binary');
response.end();
});
}).listen(3000);
console.log('server has started...');
打开浏览器:127.0.0.1:3000,就能看到图片了。
1.2 写入文件
将字符串写入到文件文件中,是非常简单的操作,使用writeFile
即可搞定:
fs.writeFile('./test.txt',data,function(err){
if(err) throw err;
console.log('写入数据成功...');
});
writeFile
方法,在没有文件时会创建文件并写入;若文件存在则内容被覆盖。
1.3 创建或文件重命名
根据writeFile
的特性,可以使用writeFile
写入空字符串的方式创建文件。
同时,fs.open
也可以创建文件:
在文件系统中,有一个fs.rename
的方法,顾名思义,对文件(文件夹)进行重命名。
特性:
将oldname文件(目录)移动至newname的路径下,并重新命名;如果oldname和newname是同一个路径,则直接进行重命名。
二、文件夹操作
通常对目录的操作比较简单一些。
2.1 读取文件夹中的文件和文件夹列表
使用fs.readdir(path,callback)
可以获取path路径下的文件和目录列表,而且只能读取直接目录下的文件和文件夹,子目录里的是获取不到的。
输出结果:
node_modules
和tmp
是文件夹,剩下的是文件,而且是获取不到node_modules
和tmp
里面的数据。获取一个目录下所有的文件,后面会讲解,稍等。
2.2 删除文件夹
使用fs.rmdir(path,callback)
可以删除文件夹,但只能删除空文件夹,如果当前路径不是文件夹或当前文件夹不为空,则删除失败;删除的为空文件夹时,可以删除成功。
如何删除不为空的目录,后面会讲解,稍等。
2.3 获取文件或文件夹的信息
fs.stat(path,callback)能够获取path路径的信息,比如创建时间,修改时间,文件大小,当前是否为文件,当前是否为文件夹等信息;如果path路径不存在,则抛出异常。
结果:
stats中的size属性就是当前文件的大小(单位:字节,除以1024即为kb),stats还有下面方法可供使用:
- stats.isFile()
- stats.isDirectory()
- stats.isBlockDevice()
- stats.isCharacterDevice()
- stats.isSymbolicLink() (only valid with fs.lstat())
- stats.isFIFO()
- stats.isSocket()
fs.stat(path,callback)
是异步执行的,对应的还有同步执行版本:fs.statSync(path)
,这个方法返回的就是fs.stats
实例。
三、综合运用
我们在上面的讲解中,还留着两个功能没实现,这里实现一下它的过程。
3.1 遍历目录中所有的文件
我们已经知道使用readdir只能获取当前目录里的文件和文件夹名称,为了获取这个目录里所有的文件名称,只能是读取当前目录里所有的文件夹里的文件。这里我们使用递归的方法,如果当前资源是文件,则进行存储,是文件夹则进行递归进一步检索,直到把所有的文件夹遍历完毕。
var result = []; // 存储获取到的文件
var stats = fs.statSync(path); // 获取当前文件的状态
if( stats.isFile() ){
result.push(path);
}else if( stats.isDirectory() ){
// 若当前路径是文件夹,则获取路径下所有的信息,并循环
var files = fs.readdirSync(path);
for(var i=0,len=files.length; i<len; i++){
var item = files[i],itempath = getLastCode(path)=='/' ? path+item : path+'/'+item; // 拼接路径
var st = fs.statSync(itempath);
if( st.isFile() ){
result.push(itempath);
}else if( st.isDirectory() ){
// 当前是文件夹,则递归检索,将递归获取到的文件列表与当前result进行拼接
var s = readDirAll( itempath );
result = result.concat( s );
}
}
}
return result;
}
console.log( readDirAll('./') );
如果想要输出一种树形的结构,就可以对当前的递归程序进行改造,比如我想要输出如下的这种结果,那么,就要分析这种结构的特点:
可以看出的规律:
我们可以再传递一个depth来表示当前目录的层级,然后计算出前面空格的数量:
depth = depth || 0; // 默认为0
var fir_code = '';
// 计算文件名称前面的字符,4个空格为1组
for(var j=0; j<depth; j++){
fir_code += ' ';
}
depth && (fir_code += '|---');
var stats = fs.statSync(path);
if( stats.isFile() ){
console.log( fir_code+path );
}else if( stats.isDirectory() ){
var files = fs.readdirSync(path);
for(var i=0,itempath = getLastCode(path)=='/' ? path+item : path+'/'+item,st = fs.statSync(itempath);
console.log( fir_code+item );
if( st.isDirectory() ){
var s = readDirAll( itempath,depth+1 );
}
}
}
}
console.log( readDirAll('./') );
3.2 删除目录
使用fs.rmdir(path)
是有局限性的,只能删除空目录,如果是个非空目录,我们可以根据上面的思路,写出一个能删除当前目录下所有的文件。递归,只要找到里面的文件夹就递归寻找,直到找到最底层,把最底层的文件删除,然后再逐级向上删除文件夹,直到删除到当前目录。
var stats = fs.statSync(path); // 获取当前文件的状态
if( stats.isFile() ){
fs.unlinkSync(path);
console.log( '删除成功: '+path );
}else if( stats.isDirectory() ){
// 若当前路径是文件夹,则获取路径下所有的信息,并循环
var files = fs.readdirSync(path);
for(var i=0,itempath = getLastCode(path)=='/' ? path+item : path+'/'+item; // 拼接路径
var st = fs.statSync(itempath);
if( st.isFile() ){
fs.unlinkSync(itempath);
console.log( '删除成功: '+itempath );
}else if( st.isDirectory() ){
// 当前是文件夹,则递归检索
rmDirAll( itempath );
}
}
// 现在可以删除文件夹
fs.rmdir(path);
console.log( '删除成功: '+path );
}
}
rmDirAll('./img');
则删除时输出的信息如下,先把内部的文件和文件夹删除干净,最后删除 ‘./img':
当然,你也可以试着实现这样的程序:
等等,都可以试着实现一下。
总结
以上就是这篇文章的全部内容了,其实文件系统里也有很多的内容需要学习,这里只是抛砖引玉,希望通过自己一点的微薄知识,能为大家指点迷津。小编还会继续分享关于node入门学习的文章,感兴趣的朋友们请继续关注编程之家。