摘要:先把當(dāng)成文件,依次查找當(dāng)前目錄下的,找到了,就返回該文件,不再繼續(xù)執(zhí)行。那么關(guān)于正確的結(jié)論是在中使用是跟的效果相同,不會(huì)因?yàn)閱?dòng)腳本的目錄不一樣而改變,在其他情況下跟效果相同,是相對(duì)于啟動(dòng)腳本所在目錄的路徑。
起因
原文收錄在我的 GitHub博客 (https://github.com/jawil/blog) ,喜歡的可以關(guān)注最新動(dòng)態(tài),大家一起多交流學(xué)習(xí),共同進(jìn)步,以學(xué)習(xí)者的身份寫博客,記錄點(diǎn)滴。
最近在學(xué)習(xí)Node.js里面的fs模塊,遇到了一個(gè)比較詭異的現(xiàn)象,踩到了坑,就是讀取當(dāng)前目錄下的一個(gè)文件,死活讀取不到,由于之前對(duì)于Node.js里面的path模塊也不太熟悉,也沒系統(tǒng)研究過,所以今天就踩了這個(gè)坑,記錄踩坑的過程,防止以后踩坑和大家也踩坑。
說一下當(dāng)時(shí)的情形:
我納悶的很半天,我明明就是讀取當(dāng)前目錄下的1.findLargest.js,為什么提示找不到這個(gè)文件,運(yùn)行了幾遍,死活找不到1.findLargest.js這個(gè)文件。
后來才發(fā)現(xiàn)是因?yàn)檫\(yùn)行這個(gè)文件不是從當(dāng)前目錄運(yùn)行了,從圖中可以看出,當(dāng)前的目錄是/Users/jawil/Desktop/nodejs/demo/ES6-lottery/syntax/nodejs,而我運(yùn)行這個(gè)腳本的目錄是/Users/jawil/Desktop/nodejs/demo/ES6-lottery;這就是問題的所在了。不過為什么運(yùn)行腳本的位置也會(huì)影響這個(gè)路徑呢,且往下看。
探索計(jì)算機(jī)不會(huì)欺騙人,一切按照規(guī)則執(zhí)行,說找不到這個(gè)文件,那肯定就是真的找不到,至于為什么找不到,那就是因?yàn)槲覀兝斫庥衅?,我最初理解?./"是當(dāng)前執(zhí)行js文件所在的文件夾的絕對(duì)路徑,然后Node.js的理解卻不是這樣的,我們慢慢往下看。
Node.js中的文件路徑大概有?__dirname,?__filename,?process.cwd(),?./?或者?../,前三個(gè)都是絕對(duì)路徑,為了便于比較,./?和?../?我們通過?path.resolve("./")來轉(zhuǎn)換為絕對(duì)路徑。
簡單說一下這幾個(gè)路徑的意思,:
__dirname: 獲得當(dāng)前執(zhí)行文件所在目錄的完整目錄名 __filename: 獲得當(dāng)前執(zhí)行文件的帶有完整絕對(duì)路徑的文件名 process.cwd():獲得當(dāng)前執(zhí)行node命令時(shí)候的文件夾目錄名 ./: 文件所在目錄
先看一看我電腦當(dāng)前的目錄結(jié)構(gòu):
syntax/ -nodejs/ -1.findLargest.js -2.path.js -3.fs.js -regs -regx.js -test.txt
在 path.js 里面我們寫這些代碼,看看輸出是什么:
const path = require("path") console.log("__dirname:", __dirname) console.log("__filename:", __filename) console.log("process.cwd():", process.cwd()) console.log("./:", path.resolve("./"))
在當(dāng)前目錄下也就是nodejs目錄運(yùn)行 node path.js,我們看看輸出結(jié)果:
__dirname: /Users/jawil/Desktop/nodejs/demo/ES6-lottery/syntax/nodejs __filename: /Users/jawil/Desktop/nodejs/demo/ES6-lottery/syntax/nodejs/2.path.js process.cwd(): /Users/jawil/Desktop/nodejs/demo/ES6-lottery/syntax/nodejs ./: /Users/jawil/Desktop/nodejs/demo/ES6-lottery/syntax/nodejs
然后在 項(xiàng)目根目錄ES6-lottery 目錄下運(yùn)行 node syntax/nodejs/2.path.js,我們?cè)賮砜纯摧敵鼋Y(jié)果:
__dirname: /Users/jawil/Desktop/nodejs/demo/ES6-lottery/syntax/nodejs __filename: /Users/jawil/Desktop/nodejs/demo/ES6-lottery/syntax/nodejs/2.path.js process.cwd(): /Users/jawil/Desktop/nodejs/demo/ES6-lottery ./: /Users/jawil/Desktop/nodejs/demo/ES6-lottery
答案顯而易見?我們可以通過上面的例子對(duì)比,暫時(shí)得出表面的結(jié)論:
__dirname: 總是返回被執(zhí)行的 js 所在文件夾的絕對(duì)路徑
__filename: 總是返回被執(zhí)行的 js 的絕對(duì)路徑
process.cwd(): 總是返回運(yùn)行 node 命令時(shí)所在的文件夾的絕對(duì)路徑
./: 跟 process.cwd() 一樣,返回 node 命令時(shí)所在的文件夾的絕對(duì)路徑
但是,我們?cè)賮砜纯催@個(gè)例子,我們?cè)谏厦娴睦蛹訋拙浯a,然后:
我們?cè)?b>1.findLargest.js先加這句代碼
exports.A = 1;
再來在剛才報(bào)錯(cuò)的3.fs.js里面加這兩句代碼看看:
const test = require("./1.findLargest.js"); console.log(test)
運(yùn)行node syntax/nodejs/3.fs.js,最后看看結(jié)果:
再次疑惑為什么都是讀取./1.findLargest.js文件,一樣的路徑,為什么require能獲取到,而readFile讀取不到呢?
于是查了不少資料,看到了一些關(guān)于require引入模塊的機(jī)制,從中學(xué)到了不少,也明白了為什么是這樣。
我們先了解一下require() 的基本用法:
下面的內(nèi)容來自require() 源碼解讀,由阮一峰翻譯自《Node使用手冊(cè)》。
我們從第(2)小條的a部分可以看出:
(2)如果 X 以 "./" 或者 "/" 或者 "../" 開頭 a. 根據(jù) X 所在的父模塊,確定 X 的絕對(duì)路徑。 b. 將 X 當(dāng)成文件,依次查找下面文件,只要其中有一個(gè)存在,就返回該文件,不再繼續(xù)執(zhí)行。
const test = require("./1.findLargest.js")按照上面規(guī)則翻譯一遍就是:
根據(jù)1.findLargest.js所在的父模塊,確定1.findLargest.js的絕對(duì)路徑為/Users/jawil/Desktop/nodejs/demo/ES6-lottery/syntax/nodejs,關(guān)于其中的尋找細(xì)節(jié)這里不做探討。
先把1.findLargest.js當(dāng)成文件,依次查找當(dāng)前目錄下的1.findLargest.js,找到了,就返回該文件,不再繼續(xù)執(zhí)行。
根據(jù)require的基本規(guī)則,對(duì)于上面出現(xiàn)的情形也就不足為奇了,更多require的機(jī)制和源碼解讀,請(qǐng)移步:
require() 源碼解讀。
那么關(guān)于 ./ 正確的結(jié)論是:
在?require()?中使用是跟?__dirname?的效果相同,不會(huì)因?yàn)閱?dòng)腳本的目錄不一樣而改變,在其他情況下跟?process.cwd()?效果相同,是相對(duì)于啟動(dòng)腳本所在目錄的路徑。
__dirname: 獲得當(dāng)前執(zhí)行文件所在目錄的完整目錄名
__filename: 獲得當(dāng)前執(zhí)行文件的帶有完整絕對(duì)路徑的文件名
process.cwd():獲得當(dāng)前執(zhí)行node命令時(shí)候的文件夾目錄名
./: 不使用require時(shí)候,./與process.cwd()一樣,使用require時(shí)候,與__dirname一樣
只有在?require()?時(shí)才使用相對(duì)路徑(./, ../) 的寫法,其他地方一律使用絕對(duì)路徑,如下:
// 當(dāng)前目錄下 path.dirname(__filename) + "/path.js"; // 相鄰目錄下 path.resolve(__dirname, "../regx/regx.js");
最后看看改過之后的結(jié)果,不會(huì)報(bào)錯(cuò)找不到文件了,不管在哪里執(zhí)行這個(gè)腳本文件,都不會(huì)出錯(cuò)了,防止以后踩坑。
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/91728.html
摘要:先把當(dāng)成文件,依次查找當(dāng)前目錄下的,找到了,就返回該文件,不再繼續(xù)執(zhí)行。那么關(guān)于正確的結(jié)論是在中使用是跟的效果相同,不會(huì)因?yàn)閱?dòng)腳本的目錄不一樣而改變,在其他情況下跟效果相同,是相對(duì)于啟動(dòng)腳本所在目錄的路徑。 起因 原文收錄在我的 GitHub博客 (https://github.com/jawil/blog) ,喜歡的可以關(guān)注最新動(dòng)態(tài),大家一起多交流學(xué)習(xí),共同進(jìn)步,以學(xué)習(xí)者的身份寫博...
摘要:前言之前在做配置時(shí)候多次用到路徑相關(guān)內(nèi)容,最近在寫項(xiàng)目的時(shí)候,有一個(gè)文件需要上傳到阿里云的功能,同時(shí)本地服務(wù)器也需要保留一個(gè)文件備份。如果返回的路徑字符串長度為零,那么他會(huì)返回一個(gè),代表當(dāng)前的文件夾。 showImg(https://segmentfault.com/img/bVbwElJ?w=480&h=204); 前言 之前在做webpack配置時(shí)候多次用到路徑相關(guān)內(nèi)容,最近在寫項(xiàng)...
摘要:簡介本文介紹了一個(gè)簡單的靜態(tài)資源服務(wù)器的實(shí)例項(xiàng)目,希望能給初學(xué)者帶來幫助。作為一個(gè)靜態(tài)資源服務(wù)器,我們當(dāng)然要加上這個(gè)功能。聲明腳本執(zhí)行類型在文件的開頭加上否則上運(yùn)行會(huì)報(bào)錯(cuò)。 簡介 本文介紹了一個(gè)簡單的靜態(tài)資源服務(wù)器的實(shí)例項(xiàng)目,希望能給Node.js初學(xué)者帶來幫助。項(xiàng)目涉及到http、fs、url、path、zlib、process、child_process等模塊,涵蓋大量常用api;...
哈。github的地址已經(jīng)更換,求starthttps://github.com/mySoul8012 繼續(xù)~Express框架 簡單介紹一下 Express事實(shí)上Node內(nèi)置的http模塊上構(gòu)建的一層抽象。理論上所有Express實(shí)現(xiàn)的功能都能用Node實(shí)現(xiàn)核心特征; 設(shè)置中間件響應(yīng)http請(qǐng)求 定義路由表,執(zhí)行不同的http請(qǐng)求 先模板傳遞參數(shù),來動(dòng)態(tài)的渲染html文件 一些網(wǎng)址 npm的...
閱讀 2735·2023-04-25 21:26
閱讀 1523·2021-11-25 09:43
閱讀 1958·2019-08-30 15:52
閱讀 939·2019-08-30 14:05
閱讀 2626·2019-08-29 16:10
閱讀 422·2019-08-29 13:48
閱讀 1868·2019-08-29 12:47
閱讀 1309·2019-08-23 18:04