首页 >web前端 >js教程 >TS 中的模块声明

TS 中的模块声明

Linda Hamilton
Linda Hamilton原创
2024-11-19 14:47:02457浏览

在这里我们学习如何编写高质量的 Typescript 声明文件。因此,您正在使用 ReactJS 进行编码并想要导入 SVG 文件,但您的 IDE/tsc 正在抱怨。

[ts] cannot find module './logo.svg'

现在是时候转向 Stackoverflow 寻求快速复制粘贴解决方案了。

但是让我们坚持一下并尝试理解这些文件以及它们对我们的作用。

什么是模块?

在 TS/JS 中,我们有很多方法来模块化用 TS/JS 编写的应用程序。目前占据主导地位的是 ESM(又名 ES 模块、ES6 模块)。我们通常通过它们的语法来了解它们:

Module declaration in TS

因此,为了回答最初的问题,在符合 ECMAScript 2015 的 TS/JS 中,任何包含 顶级导入或导出的文件都被视为模块

如果没有任何顶级导入或导出:

  1. 它们随处可见。
  2. 它们被视为脚本

为什么使用模块?

不再有全局命名空间污染。这意味着:

  1. 模块在自己的作用域内执行。
  2. 我们需要明确导出我们想要导出的任何内容。
  3. 消费者可以从不同的模块导入他们向其他人公开的内容。

命名导出和默认导出

Module declaration in TS

模块分辨率

在 TS 模块中,解析是从 import 或 require 语句中获取字符串,并确定该字符串引用哪个文件的过程。

在 TS 中我们有两种策略:

  1. 经典:
    • 默认的。
    • compilerOptions.module 不是“CommonJS”。
    • 包含在内是为了向后兼容。
  2. 节点:
    • 复制 NodeJS 在 CommonJS 模式下的工作方式。
    • 对 .ts 和 .d.ts 文件的额外检查。

顺便说一句,我们有很多地方可能会有意或无意地更改它(moduleResolution、baseUrl、paths、rootDirs)。

模块分辨率

  • 控制 TS 如何将模块说明符(导入/导出/require 语句中的字符串文字)解析为磁盘上的文件。
  • 应设置为匹配目标运行时或捆绑器使用的模块解析器

例如,如果您正在构建应用程序以利用 ESM(编译版本正在使用 ESM),那么您需要在compilerOptions.module 中选择“NodeNext”。

Module declaration in TS

  • 只要我们使用相对路径文件扩展名 TS 就可以解析它(例如 import {} from "./a.js"; // ✅ 适用于每个 moduleResolution)。
  • 如果您计划编译应用程序以使用 ESM,那么您需要指定文件的扩展名,并且不能无扩展名(例如 import {} from 'a.mjs';)。
  • 您还可以导入一个预计有index.ts文件的目录。

注意:如果在该目录中你有一个 package.json。然后 TS 使用它的 main 和 types 来获取它的类型。

在这里了解更多。

路径

所以基本上它会将导入重新映射到查找位置*s* 相对于 baseUrl(如果设置),否则映射到 tsconfig 文件。最常见的用例是路径别名:

[ts] cannot find module './logo.svg'

注意

这对于运行时来说没有任何意义。这意味着您亲爱的捆绑器或编译器需要注意将它们正确地捆绑在一起。如果你的路径指向不存在的地方,那么你的应用程序在 NodeJS 执行时将会崩溃。

提示

除此之外,您还可以利用 package.json 的导入(又名子路径导入)。

基本网址

  • 从中解析裸说明符的基目录1模块名称
  • 比从node_modules查找具有更高的优先级。
  • 使用路径时不再需要设置。

根目录

  • 许多“虚拟”目录充当单个根。
  • 然后编译器可以解析这些“虚拟”目录中的相对模块导入,就好像它们被合并到一个目录中一样。

编译后的JS

要影响发出的 JS 输出,我们可以修改:

  • 编译器选项.目标:
    • 确定将哪些 JS 功能转换为在较旧的 JavaScript 运行时中运行。
    • 由我们的应用程序需求决定,比如我将在哪个浏览器/NodeJS/Electron 上运行这个编译的 TS 代码。
  • 编译器选项.模块:
    • 系统运行时将使用哪个模块。
    • 对现代 NodeJS 项目使用“nodenext”。它读取最近的 package.json 文件并使用其“type”值来决定是否应该使用 CommonJS 还是 ESM。
    • 当您要将其与捆绑程序捆绑时,请使用“esnext”。

Module declaration in TS

回到主题

所以我们想要导入 .graphql 文件或其他扩展名,TS 不知道它们是什么样的(它无法解析它们的类型)。现在 TS 知道导入的 svg、graphql 或其他文件会是什么样子。


参考。



  1. 裸说明符:导入语句中的模块名称,不是相对或绝对路径。这些通常是项目的 node_modules 或其他配置位置中的包的名称。 ↩

以上是TS 中的模块声明的详细内容。更多信息请关注PHP中文网其他相关文章!

声明:
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn