0

0

如何在 Node.js 应用中正确使用 ES 模块组织 Express 路由

花韻仙語

花韻仙語

发布时间:2026-02-26 15:49:03

|

561人浏览过

|

来源于php中文网

原创

如何在 Node.js 应用中正确使用 ES 模块组织 Express 路由

本文详解如何将 Express 应用从 CommonJS 迁移至 ES 模块(ESM),重点解决多路由导出失败、undefined 回调错误及控制器/路由模块化设计问题,提供可直接落地的代码结构与最佳实践。

本文详解如何将 express 应用从 commonjs 迁移至 es 模块(esm),重点解决多路由导出失败、`undefined` 回调错误及控制器/路由模块化设计问题,提供可直接落地的代码结构与最佳实践。

在 Node.js 中启用 ES 模块后,Express 应用的模块组织方式需同步调整——常见错误如 Route.get() requires a callback function but got a [object Undefined],本质源于 ESM 的导出/导入语义与 CommonJS 不同:ESM 不支持动态赋值导出,且 export default 仅能声明一次;多个路由必须导出同一个 Router 实例,而非多个独立路由方法

✅ 正确的模块结构与导出方式

核心原则:路由文件应导出一个配置完成的 express.Router() 实例,而非单个路由句柄或多个未绑定的函数

Runway
Runway

Runway是一个AI创意工具平台,它提供了一系列强大的功能,旨在帮助用户在视觉内容创作、设计和开发过程中提高效率和创新能力。

下载

1. 控制器文件(controllers/shop.js):仅使用命名导出

// controllers/shop.js
export const getIndex = (req, res, next) => {
  res.render('shop/index', {
    pageTitle: 'Shop',
    path: '/',
  });
};

export const getProducts = (req, res, next) => {
  res.render('shop/product', {
    pageTitle: 'Products',
    path: '/product',
  });
};

// ❌ 删除 export default getIndex —— 命名导出已足够,重复默认导出会引发歧义

2. 路由文件(routes/shop.js):配置 Router 并导出实例

// routes/shop.js
import express from 'express';
// 推荐显式导入(更清晰、利于 tree-shaking)
import { getIndex, getProducts } from '../controllers/shop';

const router = express.Router();

// 所有路由挂载到同一 router 实例
router.get('/', getIndex);
router.get('/product', getProducts);
router.get('/cart', (req, res) => res.render('shop/cart')); // 也可内联处理

// ✅ 关键:导出整个配置好的 router 实例
export default router;

3. 入口文件(app.js):标准 ESM 导入与挂载

// app.js
import express from 'express';
import shopRouter from './routes/shop.js'; // 注意 .js 后缀(ESM 强制要求)

const app = express();

// 正确挂载:传入 router 实例
app.use(shopRouter);

// 其他中间件、端口监听等...
app.listen(3000, () => console.log('Server running on http://localhost:3000'));

⚠️ 关键注意事项

  • .js 后缀不可省略:ESM 在 import 语句中必须显式写出文件扩展名(如 ./routes/shop.js),否则会报 ERR_MODULE_NOT_FOUND。
  • 禁用 export default 冗余导出:控制器中同时存在 export const getIndex 和 export default getIndex 会导致导入行为不一致(例如 import * as ctrl from ... 无法访问默认导出),应统一使用命名导出。
  • *避免 ` as导入滥用**:虽语法合法,但会降低可读性与 IDE 支持;显式解构(import { getIndex } from ...`)更安全、更易维护。
  • 确保 package.json 配置正确
    {
      "type": "module",
      "engines": { "node": ">=18.0.0" }
    }

    Node.js ≥18 已默认支持 ESM,无需额外 flag。

✅ 总结:ESM 路由组织黄金法则

环节 正确做法 常见错误
控制器 仅用 export const fnName = (...) => {...} 混用 export default + 命名导出
路由文件 创建 Router() → 链式注册所有路由 → export default router 导出单个 router.get(...) 返回值(是 undefined)
导入方 显式导入函数或 router,带 .js 后缀 省略后缀、误用 require() 或 import() 混用

遵循此结构,你不仅能解决多路由报错问题,还能构建出高内聚、低耦合、符合现代 JavaScript 规范的 Express 应用架构。

相关文章

路由优化大师
路由优化大师

路由优化大师是一款及简单的路由器设置管理软件,其主要功能是一键设置优化路由、屏广告、防蹭网、路由器全面检测及高级设置等,有需要的小伙伴快来保存下载体验吧!

下载

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

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
json数据格式
json数据格式

JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

448

2023.08.07

json是什么
json是什么

JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

544

2023.08.23

jquery怎么操作json
jquery怎么操作json

操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

324

2023.10.13

go语言处理json数据方法
go语言处理json数据方法

本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

81

2025.09.10

Node.js后端开发与Express框架实践
Node.js后端开发与Express框架实践

本专题针对初中级 Node.js 开发者,系统讲解如何使用 Express 框架搭建高性能后端服务。内容包括路由设计、中间件开发、数据库集成、API 安全与异常处理,以及 RESTful API 的设计与优化。通过实际项目演示,帮助开发者快速掌握 Node.js 后端开发流程。

280

2026.02.10

require的用法
require的用法

require的用法有引入模块、导入类或方法、执行特定任务。想了解更多require的相关内容,可以阅读本专题下面的文章。

504

2023.11.27

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

553

2023.09.20

js正则表达式
js正则表达式

php中文网为大家提供各种js正则表达式语法大全以及各种js正则表达式使用的方法,还有更多js正则表达式的相关文章、相关下载、相关课程,供大家免费下载体验。

526

2023.06.20

Golang 实际项目案例:从需求到上线
Golang 实际项目案例:从需求到上线

《Golang 实际项目案例:从需求到上线》以真实业务场景为主线,完整覆盖需求分析、架构设计、模块拆分、编码实现、性能优化与部署上线全过程,强调工程规范与实践决策,帮助开发者打通从技术实现到系统交付的关键路径,提升独立完成 Go 项目的综合能力。

1

2026.02.26

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 9.7万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号