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

2024-08-24 23:36

本文主要是介绍TMDOG的微服务之路_07——初入微服务,NestJS微服务快速入门,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

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 快速创建微服务应用的示例。相信我们对微服务架构有了初步的了解。在下一篇博客中,我们将继续探讨更多高级的微服务架构实践,敬请期待。

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

感谢阅读!

这篇关于TMDOG的微服务之路_07——初入微服务,NestJS微服务快速入门的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1103921

相关文章

使用Node.js制作图片上传服务的详细教程

《使用Node.js制作图片上传服务的详细教程》在现代Web应用开发中,图片上传是一项常见且重要的功能,借助Node.js强大的生态系统,我们可以轻松搭建高效的图片上传服务,本文将深入探讨如何使用No... 目录准备工作搭建 Express 服务器配置 multer 进行图片上传处理图片上传请求完整代码示例

Spring LDAP目录服务的使用示例

《SpringLDAP目录服务的使用示例》本文主要介绍了SpringLDAP目录服务的使用示例... 目录引言一、Spring LDAP基础二、LdapTemplate详解三、LDAP对象映射四、基本LDAP操作4.1 查询操作4.2 添加操作4.3 修改操作4.4 删除操作五、认证与授权六、高级特性与最佳

利用Python快速搭建Markdown笔记发布系统

《利用Python快速搭建Markdown笔记发布系统》这篇文章主要为大家详细介绍了使用Python生态的成熟工具,在30分钟内搭建一个支持Markdown渲染、分类标签、全文搜索的私有化知识发布系统... 目录引言:为什么要自建知识博客一、技术选型:极简主义开发栈二、系统架构设计三、核心代码实现(分步解析

使用Python实现快速搭建本地HTTP服务器

《使用Python实现快速搭建本地HTTP服务器》:本文主要介绍如何使用Python快速搭建本地HTTP服务器,轻松实现一键HTTP文件共享,同时结合二维码技术,让访问更简单,感兴趣的小伙伴可以了... 目录1. 概述2. 快速搭建 HTTP 文件共享服务2.1 核心思路2.2 代码实现2.3 代码解读3.

Spring Boot + MyBatis Plus 高效开发实战从入门到进阶优化(推荐)

《SpringBoot+MyBatisPlus高效开发实战从入门到进阶优化(推荐)》本文将详细介绍SpringBoot+MyBatisPlus的完整开发流程,并深入剖析分页查询、批量操作、动... 目录Spring Boot + MyBATis Plus 高效开发实战:从入门到进阶优化1. MyBatis

springboot security快速使用示例详解

《springbootsecurity快速使用示例详解》:本文主要介绍springbootsecurity快速使用示例,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝... 目录创www.chinasem.cn建spring boot项目生成脚手架配置依赖接口示例代码项目结构启用s

Linux上设置Ollama服务配置(常用环境变量)

《Linux上设置Ollama服务配置(常用环境变量)》本文主要介绍了Linux上设置Ollama服务配置(常用环境变量),Ollama提供了多种环境变量供配置,如调试模式、模型目录等,下面就来介绍一... 目录在 linux 上设置环境变量配置 OllamPOgxSRJfa手动安装安装特定版本查看日志在

SpringCloud之LoadBalancer负载均衡服务调用过程

《SpringCloud之LoadBalancer负载均衡服务调用过程》:本文主要介绍SpringCloud之LoadBalancer负载均衡服务调用过程,具有很好的参考价值,希望对大家有所帮助,... 目录前言一、LoadBalancer是什么?二、使用步骤1、启动consul2、客户端加入依赖3、以服务

C++快速排序超详细讲解

《C++快速排序超详细讲解》快速排序是一种高效的排序算法,通过分治法将数组划分为两部分,递归排序,直到整个数组有序,通过代码解析和示例,详细解释了快速排序的工作原理和实现过程,需要的朋友可以参考下... 目录一、快速排序原理二、快速排序标准代码三、代码解析四、使用while循环的快速排序1.代码代码1.由快

Win32下C++实现快速获取硬盘分区信息

《Win32下C++实现快速获取硬盘分区信息》这篇文章主要为大家详细介绍了Win32下C++如何实现快速获取硬盘分区信息,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 实现代码CDiskDriveUtils.h#pragma once #include <wtypesbase