欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Node.js本地文件操作之文件拷貝與目錄遍歷的方法

 更新時(shí)間:2016年02月16日 15:51:35   投稿:goldensun  
這篇文章主要介紹了Node.js本地文件操作之文件拷貝與目錄遍歷的方法,拷貝用到了Node的文件API,遍歷則舉了一個(gè)異步API實(shí)現(xiàn)的形式(Node的一大特點(diǎn)),需要的朋友可以參考下

文件拷貝
NodeJS 提供了基本的文件操作 API,但是像文件拷貝這種高級功能就沒有提供,因此我們先拿文件拷貝程序練手。與 copy 命令類似,我們的程序需要能接受源文件路徑與目標(biāo)文件路徑兩個(gè)參數(shù)。

小文件拷貝
我們使用 NodeJS 內(nèi)置的 fs 模塊簡單實(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)容寫入目標(biāo)路徑。

豆知識: process 是一個(gè)全局變量,可通過 process.argv 獲得命令行參數(shù)。由于 argv[0] 固定等于 NodeJS 執(zhí)行程序的絕對路徑,argv[1] 固定等于主模塊的絕對路徑,因此第一個(gè)命令行參數(shù)從 argv[2] 這個(gè)位置開始。

大文件拷貝
上邊的程序拷貝一些小文件沒啥問題,但這種一次性把所有文件內(nèi)容都讀取到內(nèi)存中后再一次性寫入磁盤的方式不適合拷貝大文件,內(nèi)存會(huì)爆倉。對于大文件,我們只能讀一點(diǎn)寫一點(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)文件的只寫數(shù)據(jù)流,并且用 pipe 方法把兩個(gè)數(shù)據(jù)流連接了起來。連接起來后發(fā)生的事情,說得抽象點(diǎn)的話,水順著水管從一個(gè)桶流到了另一個(gè)桶。

遍歷目錄

遍歷目錄是操作文件時(shí)的一個(gè)常見需求。比如寫一個(gè)程序,需要找到并處理指定目錄下的所有JS文件時(shí),就需要遍歷整個(gè)目錄。

遞歸算法
遍歷目錄時(shí)一般使用遞歸算法,否則就難以編寫出簡潔的代碼。遞歸算法與數(shù)學(xué)歸納法類似,通過不斷縮小問題的規(guī)模來解決問題。以下示例說明了這種方法。

function factorial(n) {
  if (n === 1) {
    return 1;
  } else {
    return n * factorial(n - 1);
  }
}

上邊的函數(shù)用于計(jì)算 N 的階乘(N!)??梢钥吹?,當(dāng) N 大于 1 時(shí),問題簡化為計(jì)算 N 乘以 N-1 的階乘。當(dāng) N 等于 1 時(shí),問題達(dá)到最小規(guī)模,不需要再簡化,因此直接返回 1。

陷阱: 使用遞歸算法編寫的代碼雖然簡潔,但由于每遞歸一次就產(chǎn)生一次函數(shù)調(diào)用,在需要優(yōu)先考慮性能時(shí),需要把遞歸算法轉(zhuǎn)換為循環(huán)算法,以減少函數(shù)調(diào)用次數(shù)。

遍歷算法
目錄是一個(gè)樹狀結(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í),下邊這棵樹的遍歷順序是 A > B > D > E > C > F。

     A
     / \
    B  C
    / \  \
   D  E  F

同步遍歷
了解了必要的算法后,我們可以簡單地實(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í),就把文件的絕對路徑傳給回調(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)起來會(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ù)的編寫技巧,在后續(xù)章節(jié)中會(huì)詳細(xì)介紹這個(gè)。總之我們可以看到異步編程還是蠻復(fù)雜的。

相關(guān)文章

  • 快速掌握Node.js之Window下配置NodeJs環(huán)境

    快速掌握Node.js之Window下配置NodeJs環(huán)境

    快速掌握Node.js之Window下配置NodeJs環(huán)境,如何在Window下快速配置NodeJs環(huán)境,感興趣的小伙伴們可以參考一下
    2016-03-03
  • mac下安裝node.js的詳細(xì)步驟

    mac下安裝node.js的詳細(xì)步驟

    Node.js本質(zhì)上是一個(gè)JavaScript運(yùn)行時(shí)環(huán)境,它提供了在服務(wù)器端運(yùn)行JavaScript代碼的能力,這篇文章主要給大家介紹了關(guān)于mac下安裝node.js的詳細(xì)步驟,需要的朋友可以參考下
    2023-10-10
  • node.js中PC端微信小程序包解密的處理思路

    node.js中PC端微信小程序包解密的處理思路

    我們使用node.js去寫一個(gè)解碼的程序。根據(jù)上邊加密的流程,我們首先讀取加密文件,把前6個(gè)字節(jié)的固定字符串去除,這篇文章主要介紹了node.js中PC端微信小程序包解密,需要的朋友可以參考下
    2021-12-12
  • npm全局環(huán)境變量配置詳解

    npm全局環(huán)境變量配置詳解

    這篇文章主要介紹了npm全局環(huán)境變量配置詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-12-12
  • 快速解決brew安裝特定版本flow的問題

    快速解決brew安裝特定版本flow的問題

    今天小編就為大家分享一篇快速解決brew安裝特定版本flow的問題,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2018-05-05
  • node.js程序作為服務(wù)并在windows下開機(jī)自啟動(dòng)(用forever)

    node.js程序作為服務(wù)并在windows下開機(jī)自啟動(dòng)(用forever)

    這篇文章主要介紹了node.js程序作為服務(wù)并在windows下開機(jī)自啟動(dòng)的相關(guān)資料,因?yàn)閷?shí)現(xiàn)的功能比較簡單,沒有選擇功能比較強(qiáng)大的pm2,文中選擇利用了forever,需要的朋友可以參考借鑒,下面來一起看看吧。
    2017-03-03
  • Node發(fā)出HTTP POST請求的方法實(shí)例小結(jié)

    Node發(fā)出HTTP POST請求的方法實(shí)例小結(jié)

    這篇文章主要介紹了Node發(fā)出HTTP POST請求的方法,結(jié)合實(shí)例形式總結(jié)分析了三種常用的post請求操作方法,以及相關(guān)庫操作注意事項(xiàng),需要的朋友可以參考下
    2023-05-05
  • 在node環(huán)境下parse Smarty模板的使用示例代碼

    在node環(huán)境下parse Smarty模板的使用示例代碼

    這篇文章主要介紹了在node環(huán)境下parse Smarty模板的使用示例代碼,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-11-11
  • node.js將MongoDB數(shù)據(jù)同步到MySQL的步驟

    node.js將MongoDB數(shù)據(jù)同步到MySQL的步驟

    這篇文章主要給大家介紹了關(guān)于node.js將MongoDB數(shù)據(jù)同步到MySQL的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧。
    2017-12-12
  • Node.js設(shè)置CORS跨域請求中多域名白名單的方法

    Node.js設(shè)置CORS跨域請求中多域名白名單的方法

    這篇文章主要介紹了Node.js設(shè)置CORS跨域請求中多域名白名單的方法,文中通過示例代碼介紹的非常詳細(xì),相信對大家具有一定的參考價(jià)值,需要的朋友們下面來一起看看吧。
    2017-03-03

最新評論