TMDOG的微服务之路_07——初入微服务,NestJS微服务快速入门

news/2024/9/18 14:52:45/ 标签: 微服务, 架构, javascript, 云原生

TMDOG的微服务之路_07——初入微服务,NestJS微服务快速入门

博客地址:TMDOG的博客

在前几篇博客中,我们探讨了如何在 NestJS 中的一些基础功能,并可以使用NestJS实现一个简单的单体架构后端应用。本篇博客,我们将进入微服务架构,以一个简单的NestJS示例快速了解微服务架构

1. 什么是微服务

微服务架构是一种软件开发方法,将应用程序划分为多个独立的小服务,每个服务都执行特定的业务功能。这些服务可以独立部署、更新和扩展,且通常通过轻量级的通信机制(如 HTTP、TCP 等)进行交互。微服务架构的优点在于高可扩展性、灵活性和易于维护。

与传统的单体应用架构相比,微服务架构具有以下特点:

  • 模块化:将应用程序拆分为一系列小型服务,每个服务都是独立的模块,易于维护和扩展。
  • 独立部署:每个服务都可以独立部署,无需影响其他服务。
  • 松耦合:每个服务都使用独立的数据存储,相互之间松耦合,避免了单点故障。
  • 高可用性:服务可以水平扩展,以应对高流量和高并发请求。
  • 技术多样性:不同的服务可以使用不同的技术栈,例如 Java、Python、Node.js 等,充分利用各种技术的优势。

微服务架构的核心思想是将复杂的系统拆分为多个小型服务,每个服务都有一个明确的责任,从而减少系统的复杂性,并提高开发效率和灵活性。

2. NestJS 微服务快速入门

我们以一个简单的微服务架构作为示例

我们将会构建三个模块如图:
请添加图片描述

  • api-gateway API Gateway 是整个微服务架构中的入口点。它主要负责接收来自客户端的请求,并将请求分发到对应的微服务。它也可以进行身份验证、请求转发、聚合数据等操作。
  • service_1service_2 是独立的微服务,它们各自负责特定的业务逻辑。这些微服务通常根据业务需求进行拆分,每个服务专注于处理一个功能或模块。

2.1 创建项目

首先,我们先在自己的工作区创建nestjs_microservice_quickstart目录作为根目录,并且使用npm初始化

npm init -y

然后创建 microservice 目录作为微服务的目录

再使用 NestJS CLI 分别创建三个新项目,我们创建 api-gateway 作为网关服务,并分别创建两个微服务 service_1service_2。项目的基本结构如下:

nest new api-gateway
cd microservice
nest new service_1
nest new service_2
nestjs_microservice_quickstart
|-- api-gateway
|   |-- src
|         |-- 
|
|-- microservice
|   |-- service_1
|   |        |-- src
|   |
|   |-- service_2
|            |-- src

创建完成我们给每一个服务安装NestJS的微服务

npm i --save @nestjs/microservices

2.2 编码

2.2.1 编写 service_1

service_1 中,我们首先在 main.ts 中设置微服务的传输协议为 TCP,并定义其监听的端口为 3001:

import { NestFactory } from '@nestjs/core';
import { AppModule } from './app.module';
import { MicroserviceOptions, Transport } from '@nestjs/microservices';async function bootstrap() {const app = await NestFactory.createMicroservice<MicroserviceOptions>(AppModule,{transport: Transport.TCP,options: {host: '0.0.0.0',port: 3001,},},);await app.listen();
}
bootstrap();
解释
  • 与之前的单体架构项目不同的是我们创建的是微服务,并默认使用NestJS的TCP协议进行微服务之间的通信,并指定监听端口3001

接下来,我们在 app.controller.ts 中使用 MessagePattern 装饰器来处理从网关发来的消息:

import { Controller } from '@nestjs/common';
import { AppService } from './app.service';
import { MessagePattern } from '@nestjs/microservices';@Controller()
export class AppController {constructor(private readonly appService: AppService) {}@MessagePattern({ cmd: 'get_hello' })getHello(data: string): string {return this.appService.getHello(data);}
}
解释
  • 我们在MessagePattern修饰器中设置参数作为标识接收网关的消息,{ cmd: 'get_hello' }标识了该方法为getHello

app.service.ts 提供了一个简单的服务方法:

import { Injectable } from '@nestjs/common';@Injectable()
export class AppService {getHello(data: string): string {return `Hello World! This is service_1 from: ${data}`;}
}

service_2 的结构与 service_1 类似,但监听端口为 3002。

2.2.2 编写 api-gateway

api-gateway 作为微服务的入口,通过 ClientsModule 配置与 service_1service_2 的通信:

import { Module } from '@nestjs/common';
import { ClientsModule, Transport } from '@nestjs/microservices';
import { AppController } from './app.controller';
import { AppService } from './app.service';@Module({imports: [ClientsModule.register([{name: 'SERVICE_1',transport: Transport.TCP,options: { host: 'localhost', port: 3001 },},{name: 'SERVICE_2',transport: Transport.TCP,options: { host: 'localhost', port: 3002 },},]),],controllers: [AppController],providers: [AppService],
})
export class AppModule {}

app.controller.ts 中使用 ClientProxy 来与微服务通信:

import { Controller, Get, Inject, UseInterceptors } from '@nestjs/common';
import { ClientProxy } from '@nestjs/microservices';
import { LoggingInterceptor } from './common/interceptor/logger.interceptor';@Controller()
@UseInterceptors(LoggingInterceptor)
export class AppController {constructor(@Inject('SERVICE_1') private readonly service1: ClientProxy,@Inject('SERVICE_2') private readonly service2: ClientProxy,) {}@Get()async getHello(): Promise<string> {const service1Response = await this.service1.send({ cmd: 'get_hello' }, 'API Gateway').toPromise();const service2Response = await this.service2.send({ cmd: 'get_hello' }, 'API Gateway').toPromise();return `${service1Response}\n${service2Response}`;}
}

2.3 运行测试

分别进入对应的服务启动命令:

npm run start

完成编码后,我们可以分别运行 service_1service_2api-gateway,并通过浏览器或 Postman 访问 http://localhost:3000http://localhost:3000/service1http://localhost:3000/service2,测试微服务之间的通信是否正常。

运行截图:
请添加图片描述
请添加图片描述请添加图片描述

我们还可以在API网关中使用拦截器构建日志功能:

api-gateway中src/common/interceptor的logger.interceptor.ts文件下:

import { Injectable, NestInterceptor, ExecutionContext, CallHandler, Logger } from '@nestjs/common';
import { Observable  } from 'rxjs';
import { tap } from 'rxjs/operators';@Injectable()
export class LoggingInterceptor implements NestInterceptor {private readonly logger = new Logger('HTTP');intercept(context: ExecutionContext, next: CallHandler): Observable<any> {const methodKey = context.getHandler().name;const now = Date.now();return next.handle().pipe(tap(() => {this.logger.log(`method:${methodKey}-耗时: ${Date.now() - now}ms`)}));}
}

并在controller中使用

@Controller()
@UseInterceptors(LoggingInterceptor)//使用
export class AppController {constructor(@Inject('SERVICE_1') private readonly service1: ClientProxy,@Inject('SERVICE_2') private readonly service2: ClientProxy,) {}@Get()async getHello(): Promise<string> {const service1Response = await this.service1.send({ cmd: 'get_hello' }, 'API Gateway').toPromise();const service2Response = await this.service2.send({ cmd: 'get_hello' }, 'API Gateway').toPromise();return `${service1Response}\n${service2Response}`;}

请添加图片描述

结论

在本篇博客中,我们初步探讨了微服务架构,并且使用 NestJS 快速创建微服务应用的示例。相信我们对微服务架构有了初步的了解。在下一篇博客中,我们将继续探讨更多高级的微服务架构实践,敬请期待。

如有任何问题或建议,欢迎在评论区留言。

感谢阅读!


http://www.ppmy.cn/news/1516710.html

相关文章

基于改进YOLOv8的景区行人检测算法

贵向泉, 刘世清, 李立, 秦庆松, 李唐艳. 基于改进YOLOv8的景区行人检测算法[J]. 计算机工程, 2024, 50(7): 342-351. DOI: 10.19678/j.issn.10 原文链接如下&#xff1a;基于改进YOLOv8的景区行人检测算法https://www.ecice06.com/CN/rich_html/10.19678/j.issn.1000-3428.006…

解决Element-plus中Carousel(走马灯)图片无法正常加载的bug

前言&#xff1a; 最近帮助朋友解决了一个使用Element-plus中Carousel&#xff08;走马灯&#xff09;图片无法正常加载的bug&#xff0c;经过笔者的不断努力终于实现了&#xff0c;现在跟大家分享一下&#xff1a; 朋友原来的代码是这样的&#xff1a; <template><…

【计算机网络】电路交换、报文交换、分组交换

电路交换&#xff08;Circuit Switching&#xff09;&#xff1a;通过物理线路的连接&#xff0c;动态地分配传输线路资源 ​​​​

依靠 VPN 生存——探索 VPN 后利用技术

执行摘要 在这篇博文中,Akamai 研究人员强调了被忽视的 VPN 后利用威胁;也就是说,我们讨论了威胁行为者在入侵 VPN 服务器后可以用来进一步升级入侵的技术。 我们的发现包括影响 Ivanti Connect Secure 和 FortiGate VPN 的几个漏洞。 除了漏洞之外,我们还详细介绍了一组…

SpringBoot集成kafka-获取生产者发送的消息(阻塞式和非阻塞式获取)

说明 CompletableFuture对象需要的SpringBoot版本为3.X.X以上&#xff0c;需要的kafka依赖版本为3.X.X以上&#xff0c;需要的jdk版本17以上。 1、阻塞式&#xff08;等待式&#xff09;获取生产者发送的消息 生产者&#xff1a; package com.power.producer;import org.ap…

Linux的进程详解(进程创建函数fork和vfork的区别,资源回收函数wait,进程的状态(孤儿进程,僵尸进程),加载进程函数popen)

目录 什么是进程 Linux下操作进程的相关命令 进程的状态&#xff08;生老病死&#xff09; 创建进程系统api介绍&#xff1a; fork() 父进程和子进程的区别 vfork() 进程的状态补充&#xff1a; 孤儿进程 僵尸进程 回收进程资源api介绍&#xff1a; wait() waitpid…

VastBase——全局性能调优

目录 一、系统资源调优 1.内存和CPU 2.网络 3.I/O 二、查询最耗性能的SQL 三、分析作业是否被阻塞 背景&#xff1a;影响性能的因素 系统资源 数据库性能在很大程度上依赖于磁盘的I/O和内存使用情况。为了准确设置性能指标&#xff0c;用户需要了解Vastbase部署硬件的基本…

深信服研发面试经验分享

吉祥知识星球http://mp.weixin.qq.com/s?__bizMzkwNjY1Mzc0Nw&mid2247485367&idx1&sn837891059c360ad60db7e9ac980a3321&chksmc0e47eebf793f7fdb8fcd7eed8ce29160cf79ba303b59858ba3a6660c6dac536774afb2a6330#rd 《网安面试指南》http://mp.weixin.qq.com/s?…

在Spring Boot项目中集成Geth(Go Ethereum)

在Spring Boot项目中集成Geth&#xff08;Go Ethereum&#xff09;客户端&#xff0c;通常是为了与以太坊区块链进行交互。以下是一些基本的步骤和考虑因素&#xff0c;帮助你在Spring Boot应用程序中集成Geth。 安装Geth 首先&#xff0c;你需要在你的机器上安装Geth。你可以从…

k8s备份etcd3.5

一、思路 1、创建nfs存储类,用作存储备份数据<略> 2、制作用于备份的镜像文件 3、指定cronjob 二、制作镜像 ## dockerfile文件# cat Dockerfile FROM dhub.kubesre.xyz/centos:7 ADD etcdv359.tar / RUN mkdir /snapshot# docker build -t registry.k8s.io/etcd:3.…

ST表模板

P3865 【模板】ST 表 && RMQ 问题 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 思路:区间最大值&#xff0c;模板题。 int n,m; int arr[100005]; int f[100005][25]; (1<<20)1e6 void init(){ o(nlogn)for(int i1;i<…

爆改YOLOv8|利用SENetV2改进yolov8,暴力涨点

1&#xff0c;本文介绍 本文探讨了将 SENetV2 的稠密聚合层与 SE 模块结合&#xff0c;应用于 YOLOv8&#xff0c;以提升特征表达能力和目标检测性能。SENetV2 通过 Squeeze-and-Excitation&#xff08;SE&#xff09;模块优化通道和全局特征&#xff0c;从而提高分类准确率。…

UE5.4内容示例(5)UI_CommonUI - 学习笔记

https://www.unrealengine.com/marketplace/zh-CN/product/content-examples 《内容示例》是学习UE5的基础示例&#xff0c;可以用此熟悉一遍UE5的功能 UI_CommonUI可以看这个视频学习&#xff0c;此插件处于Beta状态&#xff0c;应用UI游戏方面&#xff0c;支持手柄等多输入端…

sap 开发工具 jdbc odbc 驱动 下载地址

SAP Development Tools (ondemand.com) sap 开发工具 jdbc odbc 驱动 下载地址

【系统架构设计师-2018年】综合知识-答案及详解

文章目录 【第1题】【第2~3题】【第4题】【第5~6题】【第7题】【第8题】【第9题】【第10题】【第11题】【第12题】【第13题】【第14题】【第15题】【第16~17题】【第18~21题】【第22题】【第23题】【第24题】【第25题】【第26题】【第27~28题】【第29~30题】【第31题】【第32~3…

学习前端面试知识(16)

computed和watch 参考文章vue computed 计算属性&#xff0c;有缓存功能&#xff0c;底层通过dirty来判断是否重新计算&#xff0c;只有在依赖数据发生变化时才会重新计算&#xff0c;性能更好。不能进行异步操作。缓存属性受多个属性影响&#xff0c;比如购物车商品结算函数…

OSPF-基础多区域实验

1.ENSP下载 阿里云盘分享 ⭐/*无需密钥 免费下载 安装不成功&#xff0c;可关注并私信博主*/ 2.OSPF的基础需求和规则 实验规则&#xff1a; 1.接口地址→XY.XY.XY.R /24 X:两者之间最小的 Y:两者之间最大的 R:谁的接口就是谁的编号 以R1和R2之间的连接为例&#xff0…

公司主域控服务器彻底崩溃了,蓝屏了,永久坏了!那怎么把备域提升到主域服务器呢?

一、需求描述 兄弟们&#xff0c;AD1主域控服务器彻底崩溃了&#xff0c;蓝屏了&#xff0c;永久坏了&#xff01;那怎么把AD2从备域提升到主域服务器呢&#xff1f;现在AD1主域控一直蓝屏&#xff0c;已经无法修复了。 尝试了很多方法&#xff0c;安全模式也进入不了&#xf…

系统架构分析

一、速通一图流 二、系统架构功能、作用分析 1. Furion&#xff1a;框架核心层 功能&#xff1a;这是 Furion 框架的核心层&#xff0c;通常包含框架本身的基本功能和配置。这一层应该是比较稳定的&#xff0c;不应该包含业务逻辑&#xff0c;而是提供项目其他部分需要依赖的…

《数据资产管理核心技术与应用》读书笔记-第四章:数据质量的技术实现(二)

《数据资产管理核心技术与应用》是清华大学出版社出版的一本图书&#xff0c;全书共分10章&#xff0c;第1章主要让读者认识数据资产&#xff0c;了解数据资产相关的基础概念&#xff0c;以及数据资产的发展情况。第2&#xff5e;8章主要介绍大数据时代数据资产管理所涉及的核心…