首页 > 编程语言 >NodeJS 中的设计模式

NodeJS 中的设计模式

时间:2024-03-19 09:01:22浏览次数:30  
标签:Node const NodeJS 模式 js new 设计模式

Node.js 是一个流行的 JavaScript 运行时,允许开发者使用事件驱动、非阻塞 I/O 模型构建可扩展的网络应用程序。和任何复杂的框架一样,Node.js 应用程序可以从使用成熟的设计模式中受益,以促进代码重用、可维护性和健壮性。本文将概述一些对 Node.js 开发非常有用的设计模式。

设计模式简介

设计模式是软件开发人员在编码过程中经常遇到的问题的经过验证的解决方案。它们为解决挑战提供了结构化的方法,并促进了软件架构中的最佳实践。通过整合设计模式,开发者可以创建更健壮、可维护和可扩展的代码库。

为什么设计模式在 Node.js 中很重要

Node.js 以其非阻塞事件驱动的架构而闻名,在软件设计中提出了独特的挑战和机遇。应用针对 Node.js 的设计模式可以导致更高效、更优化的应用程序。让我们探讨一些在 Node.js 生态系统中特别有价值的关键设计模式:

单例模式

单例模式确保类只有一个实例,并为其提供一个全局访问点。在 Node.js 中,模块可以被缓存并在应用程序中共享,使用单例模式可以有效地管理资源。例如,可以将数据库连接池实现为单例,以防止资源浪费。

class Database {
  constructor() {
    this.connection = null;
  }

  static getInstance() {
    if (!Database.instance) {
      Database.instance = new Database();
    }
    return Database.instance;
  }

  connect() {
    // 连接到数据库
    this.connection = 'Connected';
  }
}

const db1 = Database.getInstance();
const db2 = Database.getInstance();

console.log(db1 === db2); // true

db1.connect();

console.log(db1.connection); // 'Connected'
console.log(db2.connection); // 'Connected'

关键点是:

  • 构造函数被设为私有以防止直接实例化。
  • 静态方法 getInstance() 如果实例尚不存在,则创建一个并返回。这确保只创建一个实例。
  • 实例 db1db2 指向同一个对象。
  • db1 连接时,db2 也会获得连接,因为它们是同一个对象。

这确保只有一个数据库实例,并防止连接重复。单例模式适用于只应存在一个类实例的情况。

工厂模式

工厂模式提供了一种创建对象的方法,而无需指定将创建的对象的确切类。在 Node.js 上下文中,这可以简化对象创建,特别是在处理诸如读取文件或进行 API 调用等异步操作时。通过抽象对象创建,工厂模式增强了代码的可读性和可重用性。

class Car {
  constructor(model, price) {
    this.model = model;
    this.price = price;
  }
}

class CarFactory {
  createCar(model) {
    switch(model) {
      case 'civic':
        return new Car('Honda Civic', 20000);
      case 'accord':
        return new Car('Honda Accord', 25000);
      case 'odyssey':
        return new Car('Honda Odyssey', 30000);
      default:
        throw new Error('Unknown model');
    }
  }
}

const factory = new CarFactory();

const civic = factory.createCar('civic');
const accord = factory.createCar('accord');

console.log(civic.model); // Honda Civic
console.log(accord.model); // Honda Accord

关键点是:

  • CarFactory 类处理对象创建逻辑。
  • createCar() 方法根据型号返回一个 Car 实例。
  • 客户端代码使用工厂而不是直接构造函数调用。

这样抽象了对象创建逻辑,并允许轻松扩展支持的型号。工厂模式在存在不应与客户端代码耦合的复杂对象创建逻辑时非常有用。

观察者模式

Node.js 的事件驱动特性与观察者模式非常契合。该模式涉及维护依赖项列表(称为观察者)并通知它们任何状态更改的主体。在 Node.js 上下文中,这可以用来构建事件驱动系统,例如实时应用程序和聊天应用程序。

class Subject {
  constructor() {
    this.observers = [];
  }

  subscribe(observer) {
    this.observers.push(observer);
  }

  unsubscribe(observer) {
    this.observers = this.observers.filter(o => o !== observer);
  }

  notify(data) {
    this.observers.forEach(o => o.update(data));
  }
}

class Observer {
  constructor(name) {
    this.name = name;
  }

  update(data) {
    console.log(`${this.name} received ${data}`);
  }
}

const subject = new Subject();

const observer1 = new Observer('观察者 1');
const observer2 = new Observer('观察者 2');

subject.subscribe(observer1);
subject.subscribe(observer2);

subject.notify('你好,世界');
// 观察者 1 received 你好,世界
// 观察者 2 received 你好,世界

subject.unsubscribe(observer2);

subject.notify('再见,世界');
// 观察者 1 received 再见,世界

关键点是:

  • Subject 维护一个观察者列表。
  • 观察者通过 subscribeunsubscribe 订阅和取消订阅主体。
  • 当调用 notify() 时,主体会更新所有已订阅的观察者。

这样可以将更新发布给多个对象,而不将发布者与订阅者耦合在一起。观察者模式对于事件处理和异步工作流程非常有用。

中间件模式

Node.js 的中间件架构广泛用于处理 Web 应用程序中的请求和响应。中间件模式涉及一系列按顺序处理请求的函数。每个函数都可以在将请求或响应传递给链中的下一个函数之前修改它。这种模式增强了模块性,并允许开发者插入各种功能而不会将它们紧密耦合。

const express = require('express');
const app = express();

const logger = (req, res, next) => {
  console.log('已记录');
  next();
}

const authenticate = (req, res, next) => {
  // 认证用户
  next();
}

app.use(logger);
app.use(authenticate);

app.get('/', (req, res) => {
  res.send('你好,世界');
});

app.listen(3000);

关键点是:

  • 中间件函数 loggerauthenticate 包装路由处理程序。
  • 它们可以在路由前后执行逻辑。
  • next() 函数将控制传递给下一个中间件。
  • app.use() 在全局范围内挂载中间件。

这允许将请求处理分解为更小的可重用单元。中间件模式在 Express 和其他 Node.js 框架中非常常见,用于日志记录、身份验证等方面。

一些其他的中间件示例包括解析器、压缩、速率限制等。该模式允许以模块化方式构建请求管道。

模块模式

模块模式是 Node.js 中最基本但也是最基本的模式之一。它允许您将代码组织成单独的文件或模块,封装特定功能。

// counter.js

let count = 0;

const increment = () => {
  count++;
}

const decrement = () => {
  count--;
}

const get = () => {
  return count;
}

module.exports = {
  increment,
  decrement,
  get
};

// app.js

const counter = require('./counter');

counter.increment();
counter.increment();

console.log(counter.get()); // 2

counter.decrement();

console.log(counter.get()); // 1

关键点是:

  • 模块 counter.js 导出一些操作私有 count 变量的函数。
  • 这些函数在模块内部封装了逻辑和数据。
  • app.js 导入模块并使用公共 API。

这种模式提供了数据封装,并且只暴露了公共 API。模块模式在 Node.js 中非常常见,用于组织代码成可重用和可移植的模块。

其他一些示例包括中间件模块、实用程序库、数据访问层等。该模式有助于管理依赖关系并隐藏实现细节。

装饰器模式

装饰器动态地向对象添加新功能,而不影响其他实例。这非常适合在 Node 中扩展核心模块。

class Car {
  constructor() {
    this.price = 10000;
  }

  getPrice() {
    return this.price;
  }
}

class CarOptions {
  constructor(car) {
    this.car = car;
  }

  addGPS() {
    this.car.price += 500;
  }

  addRims() {
    this.car.price += 300;
  }
}

const basicCar = new Car();

console.log(basicCar.getPrice()); // 10000

const carWithOptions = new CarOptions(basicCar);

carWithOptions.addGPS();
carWithOptions.addRims();

console.log(carWithOptions.car.getPrice()); // 10800

关键点是:

  • CarOptions 包装了 Car 类并扩展了其行为。
  • 类似 addGPS() 的方法修改了包装的 Car 的状态。
  • 客户端具有带有附加功能的装饰后的 Car 实例。

这样可以在运行时动态扩展行为。装饰器模式对于抽象化和不必要地子类化以添加小功能非常有用。

一些其他的例子包括身份验证路由、日志包装器、缓存装饰器等。该模式提供了一种灵活的方式,以在 Node.js 应用程序中遵循开放/封闭原则。

依赖注入模式

依赖注入是一种模式,其中模块或类从外部来源接收依赖项,而不是在内部创建它们。它有助于解耦、测试和可重用性。

// service.js
class Service {
  constructor(db, logger) {
    this.db = db;
    this.logger = logger;
  }

  async getUser(userId) {
    const user = await this.db.findUserById(userId);
    this.logger.log(`Fetched user ${user.name}`);
    return user;
  }
}

// app.js
const Database = require('./database');
const Logger = require('./logger');

const db = new Database();
const logger = new Logger();

const service = new Service(db, logger);

service.getUser(1);

关键点是:

  • Service 类通过构造函数声明依赖项。
  • 调用代码注入了实际的依赖项,比如 dblogger
  • 这将 Service 与具体的依赖项解耦。

好处:

  • 模块之间的松耦合
  • 通过模拟依赖项进行更轻松的测试
  • 能够交换实现

依赖注入模式通常与 Node.js 框架(如 NestJS)一起使用。它能够更好地组织代码并提高可重用性。

Promise 模式

Promise 是 Node.js 中用于异步编程的一种模式。它们代表异步操作的最终结果。下面是一个简单的例子:

const fetchData = new Promise((resolve, reject) => {
  // 异步操作
  const data = getDataFromDatabase();

  if (data) {
    resolve(data);
  } else {
    reject('获取数据时出错');
  }
});

fetchData
  .then(data => {
    // 处理成功的数据
  })
  .catch(err => {
    // 处理错误
  });

关键点是:

  • Promise 接受一个带有 resolvereject 函数的回调。
  • 异步操作在回调内开始。
  • resolve(data) 在成功时返回数据。
  • reject(error) 在失败时返回错误。
  • 消费者使用 .then().catch() 来获取结果。

好处:

  • 处理异步结果的标准化方式
  • 能够链式和组合 Promise

Promise 对于现代 Node.js 开发至关重要,并且支持诸如 axiosfs.promises 等核心 API ,可以用来编写干净、整洁的异步代码。

应用设计模式

既然我们已经探讨了一些与 Node.js 的优势相吻合的关键设计模式,现在让我们深入了解如何有效地使用它们:

1. 理解上下文

在应用任何设计模式之前,了解应用程序的上下文非常重要。考虑诸如应用程序的需求、可扩展性需求以及您试图解决的具体挑战等因素。设计模式并不是一种适合所有场景的解决方案;它们应该根据项目的独特特性进行定制。

2. 模块化

Node.js 通过其模块系统鼓励模块化。在实现设计模式时,努力保持模块小型、专注和单一职责。这促进了代码的可重用性和可维护性,使得可以轻松地替换或增强特定功能,而不影响整个应用程序。

3. 异步模式

鉴于 Node.js 的异步特性,选择与异步编程范式相一致的设计模式至关重要。观察者模式和中间件模式等模式自然地适应了异步环境,允许开发者无缝地处理事件和异步操作。

总结

设计模式使 Node.js 开发者能够编写组织良好、灵活且健壮的代码。利用诸如工厂模式、装饰器模式和单例模式等经过验证的模式,使您能够构建易于维护和扩展的大型应用程序。理解如何应用设计原则对于掌握高级 Node 开发至关重要。

标签:Node,const,NodeJS,模式,js,new,设计模式
From: https://blog.csdn.net/Tory2/article/details/136829003

相关文章

  • 设计模式学习(一)单例模式的几种实现方式
    目录前言饿汉式懒汉式懒汉式DCLP局部静态式(Meyers'Singleton)单例模板参考文章前言单例模式,其核心目标是确保在程序运行的过程中,有且只有存在一个实例才能保证他们的逻辑正确性以及良好的效率。因此单例模式的实现思路就是确保一个类有且只有一个实例,并提供一个该实例的全局访问......
  • 使用nvm解决nodejs版本切换
    https://nvm.uihtm.com/首先电脑完全卸载电脑上已经安装的nodejs然后查看一下一下文件是否存在,存在就删除C:\ProgramFiles(x86)\NodejsC:\ProgramFiles\NodejsC:\Users\用户名\AppData\Roaming\npmC:\Users\用户名\AppData\Roaming\npm-cacheC:\Users\用户名.npmrcnvm......
  • 第四章、nodejs高级
    目录十六、知识点补充1、环境变量十六、知识点补充1、环境变量//1、nodejs自带模块化功能,一个js文件就是一个模块console.log(this===global)//false//2、console.time('start')和console.timeEnd('start')记录时间间隔console.time('start')letnum=0for(leti......
  • 安装nodejs 环境
    一、使用NVM安装Node.js1.安装nvm#$curl-o-https://raw.githubusercontent.com/nvm-sh/nvm/v0.39.2/install.sh|bash2.测试是否安装完成#command-vnvm3.列出Node.JS的所有可用版本#nvmls-remote4.列出Node.JS版本的所有最新LTS版本#nvmls......
  • [nodejs] NodeJs/NPM入门教程
    0序nodejs是运行在服务器端的js,常用于前端工程师在本地电脑、或生产环境部署调试或运行前端工程。回想起来,上次使用nodejs,还在5年前做大学毕业设计时,基于前后端分离的实践(那时,业界正在兴起前后端分离的浪潮。当然了,现在的web工程,前后端分离已是默认的技术选择了)这次重......
  • 探索设计模式的魅力:探索发布-订阅模式的深度奥秘-实现高效、解耦的系统通信
    ​......
  • Java设计模式之模板模式。
    模板模式(TemplatePattern)是Java中常用的设计模式之一,是一种行为设计模式,它定义了一个算法的骨架,将具体步骤延迟到子类中实现。模板模式使得子类可以在不改变算法结构的情况下重新定义算法中的某些步骤,下面通过Java代码看看模板模式的用法。//模板类abstractclassAbstr......
  • Java设计模式之单例模式。
    在Java中,单例模式是一种常见的设计模式,用于确保一个类只有一个实例,并提供全局访问点下面介绍七种常见的单例模式实现方式以及它们的优缺点。1.懒汉式(线程不安全)publicclassSingleton{privatestaticSingletoninstance;privateSingleton(){}publi......
  • Java设计模式之工厂模式。
    工厂模式(FactoryPattern)是Java中最常用的设计模式之一,是一种创建型设计模式,用于创建对象而不暴露创建逻辑。工厂模式包括工厂方法模式和抽象工厂模式,下面结合代码看看它的使用用法吧。//产品接口interfaceProduct{voidproduce();}//具体产品类classCon......
  • 微信小程序uniapp+vue+nodejs宝宝成长记录系统的设计与实现
    本文先通过对相关系统的调研,提出开发基于微信小程序的宝宝成长记录系统的意义,然后使用当前主流的技术进行开发,满足基于微信小程序的宝宝成长记录系统的技术要求,分析系统需要实现的功能并进行设计。梳理业务流程,并根据功能设计数据库,最后通过编码实现,介绍实现的关键算法逻辑。在......