活动介绍

Node.js模块连接模式:硬编码依赖与依赖注入

立即解锁
发布时间: 2025-08-19 01:17:25 阅读量: 1 订阅数: 4
PDF

Node.js设计模式与最佳实践

### Node.js 模块连接模式:硬编码依赖与依赖注入 #### 1. 单例模式在 Node.js 中的情况 在 Node.js 里,传统意义上的单例模式并不存在。不同的包加载同一个模块时,可能会得到不同的实例。例如,`packageA` 和 `packageB` 加载 `mydb` 模块,由于解析路径不同,会得到不同的实例。若要实现真正的单例,可使用全局变量存储实例,如 `global.db = new Database('my-app-db')`,但这种做法应尽量避免。在本文中,为简便起见,将模块导出的有状态对象称为单例,它与传统单例模式有相同的实用意图,即便于在不同组件间共享状态。 #### 2. 模块连接模式概述 在讨论了依赖和耦合的基本理论后,接下来介绍主要的模块连接模式,重点关注有状态实例的连接,因为它们是应用中最重要的依赖类型。 #### 3. 硬编码依赖 硬编码依赖是两个模块间最传统的关系。在 Node.js 中,客户端模块使用 `require()` 显式加载另一个模块时,就形成了硬编码依赖。这种方式简单有效,但对于有状态实例的硬编码依赖,会限制模块的可重用性。 ##### 3.1 使用硬编码依赖构建认证服务器 下面通过构建一个简单的认证服务器来演示硬编码依赖的使用。该服务器提供两个 HTTP API: - `POST '/login'`:接收包含用户名和密码的 JSON 对象进行认证,成功后返回 JSON Web Token (JWT)。 - `GET '/checkToken'`:从 GET 查询参数中读取令牌并验证其有效性。 为实现这个服务器,将使用 `express` 实现 Web API,使用 `levelup` 存储用户数据。 **3.1.1 db 模块** 创建 `lib/db.js` 文件,代码如下: ```javascript const level = require('level'); const sublevel = require('level-sublevel'); module.exports = sublevel( level('example-db', {valueEncoding: 'json'}) ); ``` 此模块创建一个连接到 `./example-db` 目录下的 LevelDB 数据库的实例,并使用 `sublevel` 插件进行装饰,导出的对象是数据库句柄,是一个有状态的单例。 **3.1.2 authService 模块** 实现 `lib/authService.js` 模块,代码如下: ```javascript // ... const db = require('./db'); const users = db.sublevel('users'); const tokenSecret = 'SHHH!'; exports.login = (username, password, callback) => { users.get(username, function(err, user) { // ... }); }; exports.checkToken = (token, callback) => { // ... users.get(userData.username, function(err, user) { // ... }); }; ``` 该模块实现了 `login()` 和 `checkToken()` 服务,用于验证用户凭证和令牌有效性。这里硬编码了对 `db` 模块的依赖,这意味着不修改代码就无法将 `authService` 与其他数据库实例一起使用。 **3.1.3 authController 模块** 实现 `lib/authController.js` 模块,代码如下: ```javascript const authService = require('./authService'); exports.login = (req, res, next) => { authService.login(req.body.username, req.body.password, (err, result) => { // ... } ); }; exports.checkToken = (req, res, next) => { authService.checkToken(req.query.token, (err, result) => { // ... } ); }; ``` 该模块实现了两个 Express 路由,依赖于 `authService` 模块,而 `authService` 又依赖于 `db` 模块,这种硬编码依赖会在整个应用结构中传播。 **3.1.4 app 模块** 实现应用的入口点 `app.js`,代码如下: ```javascript const express = require('express'); const bodyParser = require('body-parser'); const errorHandler = require('errorhandler'); const http = require('http'); const authController = require('./lib/authController'); const app = module.exports = express(); app.use(bodyParser.json()); app.post('/login', authController.login); app.get('/checkToken', authController.checkToken); app.use(errorHandler()); http.createServer(app).listen(3000, () => { console.log('Express server started'); }); ``` 此模块将所有组件组合在一起,硬编码了对 `authController` 有状态实例的依赖。 ##### 3.2 运行认证服务器 在运行认证服务器前,需使用 `populate_db.js` 脚本向数据库填充示例数据。然后运行以下命令启动服务器: ```bash node app ``` 可使用 REST 客户端或 `curl` 命令测试两个 Web 服务。例如,登录命令如下: ```bash curl -X POST -d '{"username": "alice", "password":"secret"}' http://localhost:3000/login -H "Content-Type: application/json" ``` 使用返回的令牌测试 `/checkToken` 服务: ```bash curl -X GET -H "Accept: application/json" http://localhost:3000/checkToken?token=<TOKEN HERE> ``` ##### 3.3 硬编码依赖的优缺点 - **优点**:直观的组织方式,易于理解和调试,每个模块可自行初始化和连接,无需外部干预。 - **缺点**:有状态实例的硬编码依赖会限制模块的可重用性和单元测试的难度。例如,`authService` 难以与其他数据库实例一起使用,且难以独立测试。不过,若使用 `require()` 加载无状态模块,如工厂函数、构造函数或无状态函数集,则不会出现这些问题。 #### 4. 依赖注入 依赖注入(DI)模式常被误解,很多人将其与框架和 DI 容器(如 Java 和 C# 的 Spring、PHP 的 Pimple)联系起来,但实际上它是一个更简单的概念。其核心思想是组件的依赖由外部实体提供,这个外部实体可以是客户端组件或全局容器。使用 DI 可提高模块的解耦性,尤其是对于依赖有状态实例的模块。 #### 5. 使用依赖注入重构认证服务器 重构模块以使用 DI 的方法是创建工厂函数,将依赖作为参数传入。 ##### 5.1 重构 db 模块 将 `lib/db.js` 模块重构为工厂函数,代码如下: ```javascript const level = require('level'); const sublevel = require('level-sublevel'); module.exports = dbName => { return sublevel( level(dbName, {val ```
corwn 最低0.47元/天 解锁专栏
赠100次下载
继续阅读 点击查看下一篇
profit 400次 会员资源下载次数
profit 300万+ 优质博客文章
profit 1000万+ 优质下载资源
profit 1000万+ 优质文库回答
复制全文

相关推荐

李_涛

知名公司架构师
拥有多年在大型科技公司的工作经验,曾在多个大厂担任技术主管和架构师一职。擅长设计和开发高效稳定的后端系统,熟练掌握多种后端开发语言和框架,包括Java、Python、Spring、Django等。精通关系型数据库和NoSQL数据库的设计和优化,能够有效地处理海量数据和复杂查询。
最低0.47元/天 解锁专栏
赠100次下载
百万级 高质量VIP文章无限畅学
千万级 优质资源任意下载
千万级 优质文库回答免费看
立即解锁

专栏目录

最新推荐

【C#事件处理机制】:深入TreeView和ListView事件,掌控用户交互的核心

# 摘要 本文旨在深入探讨C#中的事件处理机制,并结合TreeView与ListView控件的具体应用场景进行详细分析。首先概述了C#事件处理的基本概念,然后深入到TreeView和ListView控件的事件处理细节,包括事件类型的介绍、自定义事件处理程序的编写以及高级应用技巧。接着,文章展示了如何将TreeView与ListView事件集成,并应用于企业级案例,优化用户体验和提高系统效率。最后,本文展望了C#事件处理的未来发展趋势,包括在新版本中的改进,与云服务和微服务架构的融合,以及专业开发者对未来技术发展的见解和建议。 # 关键字 C#事件处理;TreeView控件;ListView控

【多源数据整合王】:DayDreamInGIS_Geometry在不同GIS格式中的转换技巧,轻松转换

![【多源数据整合王】:DayDreamInGIS_Geometry在不同GIS格式中的转换技巧,轻松转换](https://community.esri.com/t5/image/serverpage/image-id/26124i748BE03C6A81111E?v=v2) # 摘要 本论文详细介绍了DayDreamInGIS_Geometry这一GIS数据处理工具,阐述了其核心功能以及与GIS数据格式转换相关的理论基础。通过分析不同的GIS数据格式,并提供详尽的转换技巧和实践应用案例,本文旨在指导用户高效地进行数据格式转换,并解决转换过程中遇到的问题。文中还探讨了转换过程中的高级技巧、

【特征提取在心电信号情绪分析中的关键作用】:揭秘如何提取有效特征

![基于心电信号的情绪识别.rar](https://ecgwaves.com/wp-content/uploads/2017/06/exercise_ecg_st_depressions.jpg) # 摘要 本论文旨在探讨心电信号在情绪分析中的应用及其关键技术。首先介绍了心电信号的基础知识,包括其生理基础、采集技术和初步处理方法。随后,详细阐述了特征提取的理论与方法,区分了统计特征提取与高级特征提取技术,如时频分析和小波变换。在情绪分析的实际应用中,论文讨论了情绪状态与心电信号特征的关系,特征选择与优化,并通过实践案例进行了分析。最后,展望了心电信号情绪分析领域的未来发展趋势,特别是新兴技

FPGA高精度波形生成:DDS技术的顶尖实践指南

![FPGA高精度波形生成:DDS技术的顶尖实践指南](https://d3i71xaburhd42.cloudfront.net/22eb917a14c76085a5ffb29fbc263dd49109b6e2/2-Figure1-1.png) # 摘要 本文深入探讨了现场可编程门阵列(FPGA)与直接数字合成(DDS)技术的集成与应用。首先,本文介绍了DDS的技术基础和理论框架,包括其核心组件及优化策略。随后,详细阐述了FPGA中DDS的设计实践,包括硬件架构、参数编程与控制以及性能测试与验证。文章进一步分析了实现高精度波形生成的技术挑战,并讨论了高频率分辨率与高动态范围波形的生成方法。

数字通信测试理论与实践:Agilent 8960综测仪的深度应用探索

# 摘要 本文介绍了数字通信的基础原理,详细阐述了Agilent 8960综测仪的功能及其在数字通信测试中的应用。通过探讨数字信号的测试理论与调制解调技术,以及综测仪的技术指标和应用案例,本文提供了数字通信测试环境搭建与配置的指导。此外,本文深入分析了GSM/EDGE、LTE以及5G信号测试的实践案例,并探讨了Agilent 8960综测仪在高级应用技巧、故障诊断、性能优化以及设备维护与升级方面的重要作用。通过这些讨论,本文旨在帮助读者深入理解数字通信测试的实际操作流程,并掌握综测仪的使用技巧,为通信测试人员提供实用的参考和指导。 # 关键字 数字通信;Agilent 8960综测仪;调制解

虚拟助理引领智能服务:酒店行业的未来篇章

![虚拟助理引领智能服务:酒店行业的未来篇章](https://images.squarespace-cdn.com/content/v1/5936700d59cc68f898564990/1497444125228-M6OT9CELKKA9TKV7SU1H/image-asset.png) # 摘要 随着人工智能技术的发展,智能服务在酒店行业迅速崛起,其中虚拟助理技术在改善客户体验、优化运营效率等方面起到了关键作用。本文系统地阐述了虚拟助理的定义、功能、工作原理及其对酒店行业的影响。通过分析实践案例,探讨了虚拟助理在酒店行业的应用,包括智能客服、客房服务智能化和后勤管理自动化等方面。同时,

零信任架构的IoT应用:端到端安全认证技术详解

![零信任架构的IoT应用:端到端安全认证技术详解](https://img-blog.csdnimg.cn/20210321210025683.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzQyMzI4MjI4,size_16,color_FFFFFF,t_70) # 摘要 随着物联网(IoT)设备的广泛应用,其安全问题逐渐成为研究的焦点。本文旨在探讨零信任架构下的IoT安全认证问题,首先概述零信任架构的基本概念及其对Io

手机Modem协议在网络环境下的表现:分析与优化之道

![手机Modem协议开发快速上手.docx](https://img-blog.csdnimg.cn/0b64ecd8ef6b4f50a190aadb6e17f838.JPG?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBATlVBQeiInOWTpQ==,size_20,color_FFFFFF,t_70,g_se,x_16) # 摘要 Modem协议在网络通信中扮演着至关重要的角色,它不仅定义了数据传输的基础结构,还涉及到信号调制、通信流程及错误检测与纠正机制。本文首先介

【仿真模型数字化转换】:从模拟到数字的精准与效率提升

![【仿真模型数字化转换】:从模拟到数字的精准与效率提升](https://img-blog.csdnimg.cn/42826d38e43b44bc906b69e92fa19d1b.png) # 摘要 本文全面介绍了仿真模型数字化转换的关键概念、理论基础、技术框架及其在实践中的应用流程。通过对数字化转换过程中的基本理论、关键技术、工具和平台的深入探讨,文章进一步阐述了在工程和科学研究领域中仿真模型的应用案例。此外,文中还提出了数字化转换过程中的性能优化策略,包括性能评估方法和优化策略与方法,并讨论了数字化转换面临的挑战、未来发展趋势和对行业的长远意义。本文旨在为专业人士提供一份关于仿真模型数

物联网技术:共享电动车连接与控制的未来趋势

![物联网技术:共享电动车连接与控制的未来趋势](https://read.nxtbook.com/ieee/potentials/january_february_2020/assets/4cf66356268e356a72e7e1d0d1ae0d88.jpg) # 摘要 本文综述了物联网技术在共享电动车领域的应用,探讨了核心的物联网连接技术、控制技术、安全机制、网络架构设计以及实践案例。文章首先介绍了物联网技术及其在共享电动车中的应用概况,接着深入分析了物联网通信协议的选择、安全机制、网络架构设计。第三章围绕共享电动车的控制技术,讨论了智能控制系统原理、远程控制技术以及自动调度与充电管理