【第1984期】浅析 NodeJs 的几种文件路径

2020年6月22日 255点热度 0人点赞 0条评论

前言

闻道有先后,所以可以多创造一些开发的场景。今日早读文章由阿里@大果授权分享。

正文从这开始~~

一、挖坑 & 掉坑:

缘起一段这样的代码:

  1. fs.readFile('./docs/use.md', function (err, buffer) {

  2. if (err) {

  3. return console.log('error: ', err);

  4. }


  5. console.log('OK');

  6. });

本地运行时一切 OK,线上部署时却死活找不到 ./docs/use.md 这个文件,后来才发现是因为线上启动应用时不是从当前目录启动了,不过为什么启动脚本的位置也会影响这个路径呢,且往下看。

二、填坑:

Node 中的文件路径大概有 _dirname, _filename, process.cwd(), ./ 或者 ../,前三个都是绝对路径,为了便于比较,./ 和 ../ 我们通过 path.resolve('./')来转换为绝对路径。

先看一个简单的栗子:

假如我们有这样的文件结构:

  1. app/

  2. -lib/

  3. -common.js

  4. -model

  5. -task.js

  6. -test.js

在 task.js 里编写如下的代码:

  1. var path = require('path');


  2. console.log(__dirname);

  3. console.log(__filename);

  4. console.log(process.cwd());

  5. console.log(path.resolve('./'));

在 model 目录下运行 node task.js 得到的输出是:

  1. /Users/guo/Sites/learn/app/model

  2. /Users/guo/Sites/learn/app/model/task.js

  3. /Users/guo/Sites/learn/app/model

  4. /Users/guo/Sites/learn/app/model

然后在 app 目录下运行 node model/task.js,得到的输出是:

  1. /Users/guo/Sites/learn/app/model

  2. /Users/guo/Sites/learn/app/model/task.js

  3. /Users/guo/Sites/learn/app

  4. /Users/guo/Sites/learn/app

那么,不好意思不是问题来了~T_T,我们可以得出一些肤浅的结论了:

  • __dirname: 总是返回被执行的 js 所在文件夹的绝对路径

  • __filename: 总是返回被执行的 js 的绝对路径

  • process.cwd(): 总是返回运行 node 命令时所在的文件夹的绝对路径

  • ./: 跟 process.cwd() 一样、一样、一样的吗?

我明明记得在 require('../lib/common') 里一直都是各种相对路径写,也没见报什么错啊,我们还在再来个栗子吧,还是上面的结构,'model/task.js' 里的代码改成:

  1. var fs = require('fs');

  2. var common = require('../lib/common');


  3. fs.readFile('../lib/common.js', function (err, data) {

  4. if (err) return console.log(err);

  5. console.log(data);

  6. });

在 model 目录下运行 node task.js,一切 Ok,没有报错。然后在 app 目录下运行 node model/task.js,然后很果断滴报错了:

图片

那么这下问题真的都是来了,按照上面的理论,在 app 下运行时,../lib/common.js 会被转成 /Users/guo/Sites/learn/lib/common.js,这个路径显然是不存在的,但是从运行结果可以看出 require('../lib/common') 是 OK 的,只是 readFile 时报错了。

那么关于 ./ 正确的结论是:

在 require() 中使用是跟 __dirname 的效果相同,不会因为启动脚本的目录不一样而改变,在其他情况下跟 process.cwd() 效果相同,是相对于启动脚本所在目录的路径。

三、总结:

只有在 require() 时才使用相对路径(./, ../) 的写法,其他地方一律使用绝对路径,如下:

  1. // 当前目录下

  2. path.dirname(__filename) + '/test.js';

  3. // 相邻目录下

  4. path.resolve(__dirname, '../lib/common.js');

四、参考链接:

  • What is the difference between __dirname and ./ in node.js

  • nodejs API

关于本文
作者:@大果
原文:https://github.com/imsobear/blog/issues/48

图片

29770【第1984期】浅析 NodeJs 的几种文件路径

这个人很懒,什么都没留下

文章评论