Node.js 本身並沒有提供直接複製文件的 API,如果想用 Node.js 複製文件或目錄,需要藉助其他的 API 來實現。複製單個的文件可以直接用 readFile、writeFile,這樣比較簡便。如果是複製一個目錄下的所有文件,目錄下可能還包含了子目錄,那麼此時就需要用到更高級點的 API 了。
流
流是 Node.js 移動數據的方式,Node.js 中的流是可讀/可寫的,HTTP 和文件系統模塊都有用到流。在文件系統中,使用流來讀取文件的時候,對於一個大文件可能並不會一次性讀取完,而是會分好幾次讀取完,讀取的時候會響應數據事件,在文件沒讀取完的時候就可以對讀取的數據進行操作。同理,在寫入流的時候也和讀取時一樣,大文件並不會一次性寫入。這種移動數據的方式是非常高效的,尤其是對於大文件而言,使用流比等待將大文件全部讀取完再來操作文件要快得多。
管道
如果想在讀取流和寫入流的時候做完全的控制,可以使用數據事件。但對於單純的文件複製來說讀取流和寫入流可以通過管道來傳輸數據。
實際應用:
var fs = require( 'fs' ),
stat = fs.stat;
/*
* 複製目錄中的所有文件包括子目錄
* @param{ String } 需要複製的目錄
* @param{ String } 複製到指定的目錄
*/
var copy = function( src, dst ){
// 讀取目錄中的所有文件/目錄
fs.readdir( src, function( err, paths ){
if( err ){
throw err;
}
paths.forEach(function( path ){
var _src = src + '/' + path,
_dst = dst + '/' + path,
readable, writable;
stat( _src, function( err, st ){
if( err ){
throw err;
}
// 判斷是否爲文件
if( st.isFile() ){
// 創建讀取流
readable = fs.createReadStream( _src );
// 創建寫入流
writable = fs.createWriteStream( _dst );
// 通過管道來傳輸流
readable.pipe( writable );
}
// 如果是目錄則遞歸調用自身
else if( st.isDirectory() ){
exists( _src, _dst, copy );
}
});
});
});
};
// 在複製目錄前需要判斷該目錄是否存在,不存在需要先創建目錄
var exists = function( src, dst, callback ){
fs.exists( dst, function( exists ){
// 已存在
if( exists ){
callback( src, dst );
}
// 不存在
else{
fs.mkdir( dst, function(){
callback( src, dst );
});
}
});
};
// 複製目錄
exists( './src', './build', copy );