【从一个 TypeScript 报错理解 ES6 模块的三种导入方式】

从一个 TypeScript 报错理解 ES6 模块的三种导入方式

在日常开发中,我们经常遇到模块导入导出的场景。最近在处理一个项目时,遇到了一个有趣的问题:对于只有默认导出的模块,我们该使用哪种导入方式?这个问题引发了对 JavaScript 模块系统的深入思考。

问题起源

在项目中,我们遇到这样一段代码:

// 原始代码
import * as OverviewApi from '@/views/home/overview/api'// api.ts 的内容
export default {allStationInfo(data = {}) {// ... }
}

这段代码虽然能运行,但出现了 TypeScript 错误:Property 'allStationInfo' does not exist on type...。这促使我们重新思考模块导入的最佳实践。

模块化的历史演进

在深入讨论导入方式之前,我们先了解一下 JavaScript 模块化的发展历程:

  1. 早期阶段(2009年之前)

    • 没有官方模块系统
    • 主要通过全局变量和命名空间模式组织代码
    • 容易造成命名冲突和依赖混乱
  2. CommonJS时代(2009年)

    // 导出
    module.exports = { method: function() {} }
    // 导入
    const module = require('./module')
    
    • Node.js采用的模块规范
    • 同步加载,不适合浏览器环境
  3. AMD时代(2011年)

    define(['dependency'], function(dependency) {return { method: function() {} }
    })
    
    • 专为浏览器设计
    • 支持异步加载
    • 使用相对复杂
  4. ES6模块系统(2015年)

    • 官方标准化的模块系统
    • 同时支持浏览器和Node.js
    • 支持静态分析,有利于tree-shaking

三种主要的导入方式

1. 默认导入

import Api from './api'
  • 直接获取模块的默认导出
  • 最简洁的使用方式
  • 适用于模块只有一个主要导出对象的情况

2. 命名空间导入

import * as Api from './api'
  • 将所有导出(包括默认导出)收集到一个命名空间对象中
  • 默认导出会在 .default 属性下
  • 适用于模块有多个导出的情况

3. 命名导入默认导出

import { default as Api } from './api'
  • 显式地将默认导出重命名
  • 效果与默认导入相同
  • 较少使用,除非需要特别明确导入的是默认导出

使用场景分析

不同的导入方式适合不同的场景:

默认导入适用场景

  1. 单一功能模块

    // React组件
    import Button from './Button'
    // 工具类
    import axios from 'axios'
    
  2. 主要功能模块

    // 配置文件
    import config from './config'
    // API服务
    import apiService from './api'
    

命名空间导入适用场景

  1. 工具库

    // 数据可视化库
    import * as d3 from 'd3'
    // 工具函数集合
    import * as Utils from './utils'
    
  2. 类型定义

    // TypeScript类型定义
    import * as Types from './types'
    

混合使用场景

// 同时使用默认导出和具名导出
import React, { useState, useEffect } from 'react'

实际应用对比

让我们看一个具体的例子:

// api.ts
export default {method1() {},method2() {}
}// 使用方式对比
// 1. 默认导入
import Api from './api'
Api.method1()  // ✓ 推荐:简洁直观// 2. 命名空间导入
import * as ApiNamespace from './api'
ApiNamespace.default.method1()  // △ 可用但繁琐// 3. 命名导入默认导出
import { default as Api } from './api'
Api.method1()  // ✓ 作用同默认导入

TypeScript 中的考虑

在 TypeScript 项目中,选择正确的导入方式还需要考虑类型系统:

// 默认导出的类型推断通常更直接
import Api from './api'
// TypeScript 能更好地推断 Api 的类型// 命名空间导入可能需要额外的类型处理
import * as ApiNamespace from './api'
// 需要通过 .default 访问,类型推断可能不如默认导入直接

最佳实践建议

  1. 对于只有默认导出的模块

    • 使用默认导入
    • 保持代码简洁
    • 便于类型推断
  2. 对于有多个导出的模块

    • 使用具名导入
    • 或在需要时使用命名空间导入
    • 明确导入内容,便于tree-shaking
  3. 项目规范

    • 在团队中保持一致的导入方式
    • 制定清晰的模块设计规范
    • 考虑代码可维护性

结论

JavaScript 模块系统的发展给我们提供了多种导入方式的选择。在实际开发中,应该根据具体场景选择最合适的方式:

  • 理解各种导入方式的历史背景和设计初衷
  • 根据模块的导出内容选择合适的导入方式
  • 考虑团队协作和代码维护的需求
  • 注意 TypeScript 类型系统的支持情况

最重要的是,理解这些不同方式的工作原理,这样才能在遇到问题时做出正确的选择。没有绝对的对错,关键是选择最适合当前场景和团队的方式。

参考资料

  • ECMAScript 6 模块规范
  • TypeScript 模块文档
  • Node.js CommonJS 文档
  • 实际项目经验总结

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/900935.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

安徽京准:NTP网络时钟服务器功能及同步模式的介绍

安徽京准:NTP网络时钟服务器功能及同步模式的介绍 安徽京准:NTP网络时钟服务器功能及同步模式的介绍 1、NTP网络时钟服务器概念: NTP时钟服务器,表面意思是时间计量工具的服务设备,其在现代工业中是用于对客户端设备…

JMeter从入门到荒废-常见问题汇总

启动某个ThreadGroup的时候,启动不了 现象 点击start按钮的时候,结果树和汇总报告都没有任何数据。 同时,点击右上角的error log 发现有错误信息: 错误信息如下: 2025-04-09 10:03:48,009 ERROR o.a.j.g.a.ActionR…

Elasticsearch 学习规划

Elasticsearch 学习规划 明确学习目标与动机 场景化需求分析 - **S**:掌握Elasticsearch架构体系,熟练使用Elasticsearch 进行数据分析,Elasticsearch结合java 项目落地案例 - **M**:搜索和Elasticsearch相关GitHub项目 - **A**:每…

核心案例 | 湖南汽车工程职业大学无人机操控与编队技术实验室

核心案例 | 湖南汽车工程职业大学无人机操控与编队技术实验室 为满足当今无人机行业应用需求,推动无人机技术的教育与实践深度融合,北京卓翼智能科技有限公司旗下品牌飞思实验室与湖南汽车工程职业大学强强联手,共同建设无人机操控与编队技术…

【Android】Android 获取当前前台应用包名与自动化控制全流程实践笔记(适配 Android 10+)

一、前言 在 Android 系统中,获取当前运行的前台应用、返回桌面、跳转权限设置、关闭其他应用等行为,往往受到系统的严格限制。随着 Android 版本的提升(特别是 Android 10 之后,即 API 29),很多传统方法已…

Sentinel核心源码分析(上)

文章目录 前言一、客户端与Spring Boot整合二、SphU.entry2.1、构建责任链2.2、调用责任链2.2.1、NodeSelectorSlot2.2.2、ClusterBuilderSlot2.2.3、LogSlot2.2.4、StatisticSlot2.2.5、AuthoritySlot2.2.6、SystemSlot2.2.7、FlowSlot2.2.7.1、selectNodeByRequesterAndStrat…

浅谈「分词」:原理 + 方案对比 + 最佳实践

在文本搜索、自然语言处理、智能推荐等场景中,「分词」 是一个基础但至关重要的技术点。无论是用数据库做模糊查询,还是构建搜索引擎,分词都是提高效率和准确度的核心手段。 🔍 一、什么是分词? 分词(Tok…

transformers:打造的先进的自然语言处理

github地址:https://github.com/huggingface/transformers Transformers 提供了数以千计的预训练模型,支持 100 多种语言的文本分类、信息抽取、问答、摘要、翻译、文本生成。它的宗旨是让NLP 技术人易用。 Transformers 提供了便于快速下载和使用的API…

Spring Boot 集成 MongoDB 时自动创建的核心 Bean 的详细说明及表格总结

以下是 Spring Boot 集成 MongoDB 时自动创建的核心 Bean 的详细说明及表格总结: 核心 Bean 列表及详细说明 1. MongoClient 类型:com.mongodb.client.MongoClient作用: MongoDB 客户端核心接口,负责与 MongoDB 服务器建立连接、…

113. 在 Vue 3 中使用 OpenLayers 实现鼠标移动显示坐标信息

✨ 写在前面 在地图类项目开发中,一个常见需求就是:实时获取用户鼠标在地图上的经纬度坐标,并展示在地图上。 本文将通过一个简单的案例,手把手带大家在 Vue 3 项目中集成 OpenLayers 地图库,并实现以下功能&#xf…

docker配置redis容器时搭载哨兵节点的情况下配置文件docker-compose.yml示例

1.配置数据节点(主从节点) version: 3.7 services:master:image: redis:5.0.9container_name: redis-masterrestart: alwayscommand: redis-server --appendonly yesports:- 6379:6379slave1:image: redis:5.0.9container_name: redis-slave1restart: a…

C++建造者模式进化论

还在为 C 对象那 长得令人发指 的构造函数参数列表抓狂吗?🤯 是不是经常在 int hp, int mp, int strength, int faith... 这样的参数“连连看”中迷失自我,一不小心就把法力值传给了血量,或者力量值填到了信仰栏?&…

在Ubuntu内网环境中为Gogs配置HTTPS访问(通过Apache反向代理使用IP地址)

一、准备工作 确保已安装Gogs并运行在HTTP模式(默认端口3000) 确认服务器内网IP地址(如192.168.1.100) 二、安装Apache和必要模块 sudo apt update sudo apt install apache2 -y sudo a2enmod ssl proxy proxy_http rewrite headers 三、创建SSL证书 1. 创建证书存储目录…

数据中台、BI业务访谈(二):组织架构梳理的坑

这是数据中台、BI业务访谈系列的第二篇文章,在上一篇文章中,我重点介绍了在给企业的业务部门、高层管理做业务访谈之前我们要做好行业、业务知识的功课。做好这些功课之后,就到了实际的访谈环节了。 业务访谈关键点 那么在具体业务访谈的时…

spark集群,Stand alone,Hadoop集群有关启动问题

你的问题是因为 start-all.sh 是 Hadoop 的启动脚本(用于启动 HDFS 和 YARN),而不是 Spark 的启动脚本。而你已经通过 start-cluster.sh 启动了 Hadoop 相关服务(HDFS/YARN),再次执行 start-all.sh 会导致服…

Kotlin 通用请求接口设计:灵活处理多样化参数

在 Kotlin 中设计一个通用的 ControlParams 类来处理不同的控制参数,有几种常见的方法:方案1:使用密封类(Sealed Class) sealed class ControlParamsdata class LightControlParams(val brightness: Int,val color: S…

aspark 配置2

编写Hadoop集群启停脚本 1.建立新文件,编写脚本程序 在hadoop101中操作,在/root/bin下新建文件:myhadoop,输入如下内容: 2.分发执行权限 保存后退出,然后赋予脚本执行权限 [roothadoop101 ~]$ chmod x /r…

Webstorm 使用搜不到node_modules下的JS内容 TS项目按Ctrl无法跳转到函数实现

将node_modules标记为不排除,此时要把内存改大,不然webstorm中途建立索引时,会因为内存不足,导致索引中途停止,造成后续搜索不出来 更改使用内存设置 内存调为4096 若出现搜不出来js内容时,请直接重启下该项…

vue-element-plus-admin的安装

文档链接:开始 | vue-element-plus-admin 之前尝试按照官方文档来安装,运行npm run dev命令却不能正常打开访问浏览器,换一个方式 首先在目录下打开命令窗口 1、克隆项目 从 GitHub 获取代码 # clone 代码 git clone https://github.com…

【windows10】基于SSH反向隧道公网ip端口实现远程桌面

【windows10】基于SSH反向隧道公网ip端口实现远程桌面 1.背景2.SSH反向隧道3.远程连接电脑 1.背景 ‌Windows 10远程桌面协议的简称是RDP(Remote Desktop Protocol)‌。 RDP是一种网络协议,允许用户远程访问和操作另一台计算机。 远程桌面功…