Node.js本地文件操作之文件拷貝與目錄遍歷的方法
文件拷貝
NodeJS 提供了基本的文件操作 API,但是像文件拷貝這種高級(jí)功能就沒(méi)有提供,因此我們先拿文件拷貝程序練手。與 copy 命令類(lèi)似,我們的程序需要能接受源文件路徑與目標(biāo)文件路徑兩個(gè)參數(shù)。
小文件拷貝
我們使用 NodeJS 內(nèi)置的 fs 模塊簡(jiǎn)單實(shí)現(xiàn)這個(gè)程序如下。
var fs = require('fs');
function copy(src, dst) {
fs.writeFileSync(dst, fs.readFileSync(src));
}
function main(argv) {
copy(argv[0], argv[1]);
}
main(process.argv.slice(2));
以上程序使用 fs.readFileSync 從源路徑讀取文件內(nèi)容,并使用 fs.writeFileSync 將文件內(nèi)容寫(xiě)入目標(biāo)路徑。
豆知識(shí): process 是一個(gè)全局變量,可通過(guò) process.argv 獲得命令行參數(shù)。由于 argv[0] 固定等于 NodeJS 執(zhí)行程序的絕對(duì)路徑,argv[1] 固定等于主模塊的絕對(duì)路徑,因此第一個(gè)命令行參數(shù)從 argv[2] 這個(gè)位置開(kāi)始。
大文件拷貝
上邊的程序拷貝一些小文件沒(méi)啥問(wèn)題,但這種一次性把所有文件內(nèi)容都讀取到內(nèi)存中后再一次性寫(xiě)入磁盤(pán)的方式不適合拷貝大文件,內(nèi)存會(huì)爆倉(cāng)。對(duì)于大文件,我們只能讀一點(diǎn)寫(xiě)一點(diǎn),直到完成拷貝。因此上邊的程序需要改造如下。
var fs = require('fs');
function copy(src, dst) {
fs.createReadStream(src).pipe(fs.createWriteStream(dst));
}
function main(argv) {
copy(argv[0], argv[1]);
}
main(process.argv.slice(2));
以上程序使用 fs.createReadStream 創(chuàng)建了一個(gè)源文件的只讀數(shù)據(jù)流,并使用 fs.createWriteStream 創(chuàng)建了一個(gè)目標(biāo)文件的只寫(xiě)數(shù)據(jù)流,并且用 pipe 方法把兩個(gè)數(shù)據(jù)流連接了起來(lái)。連接起來(lái)后發(fā)生的事情,說(shuō)得抽象點(diǎn)的話(huà),水順著水管從一個(gè)桶流到了另一個(gè)桶。
遍歷目錄
遍歷目錄是操作文件時(shí)的一個(gè)常見(jiàn)需求。比如寫(xiě)一個(gè)程序,需要找到并處理指定目錄下的所有JS文件時(shí),就需要遍歷整個(gè)目錄。
遞歸算法
遍歷目錄時(shí)一般使用遞歸算法,否則就難以編寫(xiě)出簡(jiǎn)潔的代碼。遞歸算法與數(shù)學(xué)歸納法類(lèi)似,通過(guò)不斷縮小問(wèn)題的規(guī)模來(lái)解決問(wèn)題。以下示例說(shuō)明了這種方法。
function factorial(n) {
if (n === 1) {
return 1;
} else {
return n * factorial(n - 1);
}
}
上邊的函數(shù)用于計(jì)算 N 的階乘(N!)??梢钥吹?,當(dāng) N 大于 1 時(shí),問(wèn)題簡(jiǎn)化為計(jì)算 N 乘以 N-1 的階乘。當(dāng) N 等于 1 時(shí),問(wèn)題達(dá)到最小規(guī)模,不需要再簡(jiǎn)化,因此直接返回 1。
陷阱: 使用遞歸算法編寫(xiě)的代碼雖然簡(jiǎn)潔,但由于每遞歸一次就產(chǎn)生一次函數(shù)調(diào)用,在需要優(yōu)先考慮性能時(shí),需要把遞歸算法轉(zhuǎn)換為循環(huán)算法,以減少函數(shù)調(diào)用次數(shù)。
遍歷算法
目錄是一個(gè)樹(shù)狀結(jié)構(gòu),在遍歷時(shí)一般使用深度優(yōu)先+先序遍歷算法。深度優(yōu)先,意味著到達(dá)一個(gè)節(jié)點(diǎn)后,首先接著遍歷子節(jié)點(diǎn)而不是鄰居節(jié)點(diǎn)。先序遍歷,意味著首次到達(dá)了某節(jié)點(diǎn)就算遍歷完成,而不是最后一次返回某節(jié)點(diǎn)才算數(shù)。因此使用這種遍歷方式時(shí),下邊這棵樹(shù)的遍歷順序是 A > B > D > E > C > F。
A
/ \
B C
/ \ \
D E F
同步遍歷
了解了必要的算法后,我們可以簡(jiǎn)單地實(shí)現(xiàn)以下目錄遍歷函數(shù)。
function travel(dir, callback) {
fs.readdirSync(dir).forEach(function (file) {
var pathname = path.join(dir, file);
if (fs.statSync(pathname).isDirectory()) {
travel(pathname, callback);
} else {
callback(pathname);
}
});
}
可以看到,該函數(shù)以某個(gè)目錄作為遍歷的起點(diǎn)。遇到一個(gè)子目錄時(shí),就先接著遍歷子目錄。遇到一個(gè)文件時(shí),就把文件的絕對(duì)路徑傳給回調(diào)函數(shù)。回調(diào)函數(shù)拿到文件路徑后,就可以做各種判斷和處理。因此假設(shè)有以下目錄:
- /home/user/
- foo/
x.js
- bar/
y.js
z.css
使用以下代碼遍歷該目錄時(shí),得到的輸入如下。
travel('/home/user', function (pathname) {
console.log(pathname);
});
/home/user/foo/x.js /home/user/bar/y.js /home/user/z.css
異步遍歷
如果讀取目錄或讀取文件狀態(tài)時(shí)使用的是異步API,目錄遍歷函數(shù)實(shí)現(xiàn)起來(lái)會(huì)有些復(fù)雜,但原理完全相同。travel函數(shù)的異步版本如下。
function travel(dir, callback, finish) {
fs.readdir(dir, function (err, files) {
(function next(i) {
if (i < files.length) {
var pathname = path.join(dir, files[i]);
fs.stat(pathname, function (err, stats) {
if (stats.isDirectory()) {
travel(pathname, callback, function () {
next(i + 1);
});
} else {
callback(pathname, function () {
next(i + 1);
});
}
});
} else {
finish && finish();
}
}(0));
});
}
這里不詳細(xì)介紹異步遍歷函數(shù)的編寫(xiě)技巧,在后續(xù)章節(jié)中會(huì)詳細(xì)介紹這個(gè)??傊覀兛梢钥吹疆惒骄幊踢€是蠻復(fù)雜的。
- 使用nodeJS中的fs模塊對(duì)文件及目錄進(jìn)行讀寫(xiě),刪除,追加,等操作詳解
- 純異步nodejs文件夾(目錄)復(fù)制功能
- nodejs 遞歸拷貝、讀取目錄下所有文件和目錄
- Node.js fs模塊(文件模塊)創(chuàng)建、刪除目錄(文件)讀取寫(xiě)入文件流的方法
- Node.JS枚舉統(tǒng)計(jì)當(dāng)前文件夾和子目錄下所有代碼文件行數(shù)
- Node.JS循環(huán)刪除非空文件夾及子目錄下的所有文件
- node.js基于fs模塊對(duì)系統(tǒng)文件及目錄進(jìn)行讀寫(xiě)操作的方法詳解
- Node.js查找當(dāng)前目錄下文件夾實(shí)例代碼
- Node.js實(shí)現(xiàn)在目錄中查找某個(gè)字符串及所在文件
- NodeJS 創(chuàng)建目錄和文件的方法實(shí)例分析
相關(guān)文章
node+experss實(shí)現(xiàn)爬取電影天堂爬蟲(chóng)
本文給大家分享的是node+experss制作爬蟲(chóng)的第二篇,我們來(lái)爬取電影天堂最新更新的電影迅雷下載鏈接,有需要的小伙伴可以參考下2016-11-11
基于nodejs res.end和res.send的區(qū)別
今天小編就為大家分享一篇基于nodejs res.end和res.send的區(qū)別,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-05-05
一文教會(huì)你從Windows中完全刪除node.js
作為新手nodejs卸載后安裝就總出錯(cuò),下面這篇文章主要給大家介紹了關(guān)于如何從Windows中完全刪除node.js的相關(guān)資料,文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-08-08
手寫(xiě)Node靜態(tài)資源服務(wù)器的實(shí)現(xiàn)方法
這篇文章主要介紹了手寫(xiě)Node靜態(tài)資源服務(wù)器的實(shí)現(xiàn)方法,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-03-03
node.js中的fs.statSync方法使用說(shuō)明
這篇文章主要介紹了node.js中的fs.statSync方法使用說(shuō)明,本文介紹了fs.statSync的方法說(shuō)明、語(yǔ)法、接收參數(shù)、使用實(shí)例和實(shí)現(xiàn)源碼,需要的朋友可以參考下2014-12-12
Node.js調(diào)用fs.renameSync報(bào)錯(cuò)(Error: EXDEV, cross-device link not
這篇文章主要介紹了Node.js調(diào)用fs.renameSync報(bào)錯(cuò)(Error: EXDEV, cross-device link not permitted),非常具有實(shí)用價(jià)值,需要的朋友可以參考下2017-12-12
node.js利用mongoose獲取mongodb數(shù)據(jù)的格式化問(wèn)題詳解
這篇文章主要給大家介紹了關(guān)于node.js利用mongoose獲取mongodb數(shù)據(jù)的格式化問(wèn)題,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)把。2017-10-10
nodejs實(shí)現(xiàn)HTTPS發(fā)起POST請(qǐng)求
這篇文章主要介紹了nodejs實(shí)現(xiàn)HTTPS發(fā)起POST請(qǐng)求的實(shí)例代碼,非常的簡(jiǎn)單實(shí)用,有需要的小伙伴可以參考下。2015-04-04
使用Node.js搭建靜態(tài)資源服務(wù)詳細(xì)教程
這篇文章主要介紹了使用Node.js搭建靜態(tài)資源服務(wù)器,需要的朋友可以參考下2017-08-08

