NodeJS 之 fs 模块(路径动态拼接问题)

导读:本篇文章讲解 NodeJS 之 fs 模块(路径动态拼接问题),希望对大家有帮助,欢迎收藏,转发!站点地址:www.bmabk.com

参考

项目 参考
搜索引擎 Bing
哔哩哔哩 黑马程序员

描述

项目 描述
操作系统 Windows 10 专业版
NodeJS 18.13.0

问题

描述

在使用 NodeJS 的 fs 模块处理文件时,你可能经常在路径中使用 ./../ 来指明某个文件(夹)位于当前工作目录或其父目录中。使用相对路径指定文件的位置相比使用绝对路径来指定文件的位置要方便许多,但这样可能会导致意外事故的发生。

准备工作

content.txt

这是 content.txt 文件中的内容,我们稍后将通过 fs 模块读取该文件。

隐约雷鸣,阴霾天空。但盼风雨来,能留你在此。
隐约雷鸣,阴霾天空。即使无天雨,我亦留此地。

main.js

这是 main.js 文件中的内容,稍后我将通过 NodeJS 执行该文件以读取 content.txt 文件中的内容。

const fs = require('fs');

// 尝试读取当前工作目录下的 content.txt 文件
fs.readFile('./content.txt', 'utf-8', (err, data) => {
    if(err){
        console.log('【文件读取失败】')
    }else{
        console.log(data)
    }
})

注:

  1. content.txtmain.js 文件位于同一目录之下。
  2. content.txtmain.js 文件所在的路径分别为:
    • C:\Users\36683\Template\content.txt
    • C:\Users\36683\Template\main.js

复现

前奏

进入 main.js 所在的目录下后,在终端中输入如下命令以执行 main.js 文件中的内容。

node main.js

终端将输出如下内容:

隐约雷鸣,阴霾天空。但盼风雨来,能留你在此。
隐约雷鸣,阴霾天空。即使无天雨,我亦留此地。

惊雷

首先,让我们使用如下命令进入当前目录的父级目录中:

cd ..

执行如下命令以读取 content.txt 文件中的内容:

node Template/main.js

打印结果为:

【文件读取失败】

再探

由于我们使用 fs 模块捕获了错误,所以错误信息并没有打印出来。我们需要获得错误信息以对该问题产生的原因有更准确的了解。

main.js 文件中的内容修改为:

const fs = require('fs');

// 尝试读取当前工作目录下的 content.txt 文件
fs.readFile('./content.txt', 'utf-8', (err, data) => {
    if(err){
        console.log('【文件读取失败】');
        // 打印错误信息
        console.log(err.message);
    }else{
        console.log(data)
    }
})

再次执行如下命令以读取 content.txt 文件中的内容:

node Template/main.js

打印结果:

【文件读取失败】
ENOENT: no such file or directory, open ‘C:\Users\36683\content.txt’

分析

content.txt 文件的路径应该为:

C:\Users\36683\Template\content.txt

可再错误信息中得到的却是:

C:\Users\36683\content.txt

这是因为,NodeJS 在执行 JavaScript 文件时,会将其中的相对路径与执行命令时所在的工作目录进行拼接。

解决

前面你所认识到的问题便是路径拼接问题,该问题的解决方法有两种:

  1. 使用绝对路径而非相对路径指定计算机中的文件。
  2. 通过 NodeJS 来获取被 NodeJS 执行的文件所在的文件夹的路径与目标文件(将要读取的文件)的相对路径进行拼接得到目标文件的绝对路径。

第一种解决方式相信各位都懂;第二种方式更具灵活性,不用担心将项目文件(包含 content.txtmain.js 文件的文件夹)转移到其他计算机后无法正常使用。

__dirname

__dirname 是 NodeJS 中内置的一个变量,保存着该文件所在的目录的路径。

首先,让我们在终端中使用如下命令切换工作目录至 main.js 文件所在的文件夹中。

cd Template

进来该目录后,将 main.js 修改为如下内容。

const fs = require('fs');

// 尝试读取当前工作目录下的 content.txt 文件
fs.readFile('./content.txt', 'utf-8', (err, data) => {
    if(err){
        console.log('【文件读取失败】');
        console.log(err.message);
    }else{
        console.log(__dirname)
    }
})

执行结果为:

C:\Users\36683\Template

path 模块

在使用 __dirname 获取到被 NodeJS 执行的文件所在的文件夹的路径后,我们需要将其与目标文件(将要读取的文件)的相对路径进行拼接。

路径的拼接尽量不要使用像拼接字符串那样使用 + 号进行拼接,更好的方法是使用 NodeJS 内置的 path 模块中 path.join() 来进行拼接。

path.join()

Windows 使用的文件分割符为 \ ,当然你也可以使用 / ,但这并不规范;而在 LinuxMacOS 中,操作系统使用的文件分割符均为 / ,在这些操作系统中,\ 将被理解为转义字符的前缀。

为了 提高程序在不同平台中的兼容性 ,我们需要正确的使用文件分隔符,而 path.join() 很好的解决了这个问题。

path.join() 方法使用特定于平台的分隔符作为定界符将所有给定的 path 片段连接在一起,然后规范化生成的路径。

path.join([...paths])
项目 描述
paths paths 为一个路径中所有被符号分割的文件(夹)名或盘符,你需要将这些片段按照一定的顺序传递给 path.join() 函数。

举个栗子:

const path = require('path');

result = path.join('c', 'TwoMoons', 'RedHeart');
console.log(result)

打印结果:

C:\Users\36683\Template

注:

由打印结果可以发现,path.join() 将我们传递给其的 c 转换为 C

解决

main.js 修改为如下内容后,切换工作目录为当前工作目录的父级目录。

const fs = require('fs');
const path = require('path');

const target = path.join(__dirname, 'content.txt')

// 尝试读取当前工作目录下的 content.txt 文件
fs.readFile(target, 'utf-8', (err, data) => {
    if(err){
        console.log('【文件读取失败】');
        console.log(err.message);
    }else{
        console.log(data)
    }
})

执行结果:

隐约雷鸣,阴霾天空。但盼风雨来,能留你在此。
隐约雷鸣,阴霾天空。即使无天雨,我亦留此地。

至此,路径动态拼接问题已成功解决。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

文章由极客之音整理,本文链接:https://www.bmabk.com/index.php/post/84019.html

(0)
小半的头像小半

相关推荐

极客之音——专业性很强的中文编程技术网站,欢迎收藏到浏览器,订阅我们!