一篇文章学会如何在 NestJS 中使用 Redis 并基于 Redis 实现接口访问限频率

前言

在处理高频数据操作和大规模并发请求的场合,我们需要一种机制能够快速读取和缓存数据,这时 Redis 就闪亮登场了。Redis 是一个开源的内存中数据结构存储系统,它可以用作数据库、缓存和消息中间件。

NestJS 是一个灵活且模块化的Node.js框架,它借鉴了Angular的设计哲学,提出了控制器、提供者和模块的概念,这为我们集成和使用Redis提供了便利。本文介绍如何在NestJS框架中集成Redis,并通过实际案例来展示使用Redis的优势。

集成步骤

一、 安装必要的包

在开始之前,我们需要在NestJS项目中安装 Redis 和 NestJS 对应的 Redis 模块。

npm install redis ioredis @nestjs-modules/ioredis @nestjs/common

这里我们使用 ioredis,因为它是一个健壮的、功能全面的 Redis 客户端,与 nest-modules相关联。

二、创建 Redis 模块

接下来,我们需要在 NestJS 应用中创建一个 Redis 模块。

// redis.module.ts
import { Module } from '@nestjs/common';
import { RedisModule } from '@nestjs-modules/ioredis';@Module({imports: [RedisModule.forRoot({config: {host: 'localhost', // Redis 服务器地址port: 6379,       // Redis 端口password: 'your_password', // 如果有设置密码的话db: 0,           // 如果你需要使用特定的数据库的话},}),],
})
export class RedisCacheModule {}

在这个模块中,我们导入并配置了 Redis。我们需要确保这些配置与你的 Redis 服务器设置匹配。

三、在服务中使用 Redis

配置好模块后,我们可以在服务中注入 Redis 客户端,并开始实现业务逻辑。

// app.service.ts
import { Injectable } from '@nestjs/common';
import { InjectRedis, Redis } from '@nestjs-modules/ioredis';@Injectable()
export class AppService {constructor(@InjectRedis() private readonly redis: Redis) {}async getHello(): Promise<string> {// 使用 Redis 设置值await this.redis.set('hello', 'Hello from Redis!');// 使用 Redis 获取值return this.redis.get('hello');}// ... 其他业务逻辑 ...
}

在这个服务中,我们注入了 Redis 客户端,并在 getHello方法中演示了如何设置和获取缓存。

四、控制器 Controller 调用服务

最后,让我们来创建一个控制器调用这个服务。

// app.controller.ts
import { Controller, Get } from '@nestjs/common';
import { AppService } from './app.service';@Controller()
export class AppController {constructor(private readonly appService: AppService) {}@Get()getHello(): Promise<string> {return this.appService.getHello();}
}

控制器非常简单,它使用 @Get()装饰器来处理根路由的 GET 请求,并调用 appService中的 getHello方法。

功能实现

通过将Redis集成到NestJS中,我们可以实现以下需求:

  1. 数据缓存:存储经常查询的数据提高处理速度。
  2. 会话管理:用Redis存储session信息,实现无状态的负载均衡。
  3. 排行榜系统:Redis的Sorted Set非常适合做排名操作。
  4. 发布/订阅:利用 Redis 实现分工系统,用于消息的异步处理。
  5. 限流:通过Redis的INCR和EXPIRE命令来实现API的限流。
  6. 作业队列:使用 Redis 的列表结构实现一个简单的作业队列。

实战:接口访问限制频率

为了控制访问速率并避免应用被过多的请求淹没,我们可以利用 Redis 来为我们的 API 接口实现限流功能。下面是一个简化的例子:

步骤一:实现限流拦截器

// api-rate-limiter.interceptor.ts
import {Injectable,NestInterceptor,ExecutionContext,CallHandler,HttpException,HttpStatus,
} from '@nestjs/common';
import { Observable } from 'rxjs';
import { InjectRedis, Redis } from '@nestjs-modules/ioredis';
import { tap } from 'rxjs/operators';@Injectable()
export class ApiRateLimiterInterceptor implements NestInterceptor {constructor(@InjectRedis() private readonly redis: Redis) {}async intercept(context: ExecutionContext, next: CallHandler): Promise<Observable<any>> {const key = 'rate-limit:' + context.switchToHttp().getRequest().ip;const currentRequestCount = await this.redis.incr(key);if (currentRequestCount === 1) {// 设置 key 的超时时间await this.redis.expire(key, 60); // 限流周期为 60 秒}if (currentRequestCount > 10) {throw new HttpException('Too many requests', HttpStatus.TOO_MANY_REQUESTS);}return next.handle().pipe(tap(() => {// 在响应完成后,你可以在这里执行一些操作。}),);}
}

要实现的这个拦截器会检查每个 IP 地址每分钟发出的请求,并且如果请求超过10次,将会抛出 429 Too Many Requests 错误。

步骤二:项目中使用限流拦截器

将这个拦截器中间件引入到你的应用中,可以在对应的控制器或全局应用中注册。

// 在主模块中全局注册
import { Module, NestModule, MiddlewareConsumer } from '@nestjs/common';
import { ApiRateLimiterInterceptor } from './api-rate-limiter.interceptor';@Module({// ...
})
export class AppModule implements NestModule {configure(consumer: MiddlewareConsumer) {consumer.apply(ApiRateLimiterInterceptor).forRoutes('*'); // 应用到所有的路由}
}

或者在特定的控制器上使用:

// app.controller.ts
import { Controller, UseInterceptors, Get } from '@nestjs/common';
import { ApiRateLimiterInterceptor } from './api-rate-limiter.interceptor';@Controller()
@UseInterceptors(ApiRateLimiterInterceptor)
export class AppController {// ...
}

总结

通过在 NestJS 中集成 Redis,我们不仅可以实现上述功能,还能设计更复杂的系统,如分布式系统通信、实时数据分析等等。Redis 的快速、灵活性加上 NestJS 的架构,能够让你的 Web 应用性能达到一个新的层次。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/597650.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

frp透传软件最新toml格式的配置文件的使用

frp软件在0.52.0版本开始支持 toml格式的配置文件了&#xff0c;并将在后继某个版本开始取消对 ini配置格式的支持。这里做一下新旧配置文件的比较。 一、frps 服务端配置文件的变化 frps.ini cat /etc/frp/frps.ini[common] bind_port7000 vhost_http_port8080 vhost_https…

JVM之内存模型带参数

Spring Boot程序的JVM参数设置格式(Tomcat启动直接加在bin目录下catalina.sh文件里)&#xff1a; java ‐Xms2048M ‐Xmx2048M ‐Xmn1024M ‐Xss512K ‐XX:MetaspaceSize256M ‐XX:MaxMetaspaceSize256M ‐jar xxxxxx.jar-Xss&#xff1a;每个线程的栈大小 -Xms&#xff1a;设置…

关于“Python”的核心知识点整理大全61

目录 注意 20.1.4 使用 jumbotron 设置主页的样式 index.html 20.1.5 设置登录页面的样式 login.html 20.1.6 设置 new_topic 页面的样式 new_topic.html 20.1.7 设置 topics 页面的样式 topics.html 元素&#xff0c;让它们在页面上显得大些&#xff08;见2&#xf…

imgaug库指南(三):从入门到精通的【图像增强】之旅

引言 在深度学习和计算机视觉的世界里&#xff0c;数据是模型训练的基石&#xff0c;其质量与数量直接影响着模型的性能。然而&#xff0c;获取大量高质量的标注数据往往需要耗费大量的时间和资源。正因如此&#xff0c;数据增强技术应运而生&#xff0c;成为了解决这一问题的…

prometheus与zabbix监控的对比介绍

一、普米与zabbix基本介绍 1、prometheus介绍 Prometheus的基本原理是Prometheus Server通过HTTP周期性抓取被监控组件的监控数据&#xff0c;任意组件只要提供对应的HTTP接口并且符合Prometheus定义的数据格式&#xff0c;就可以接入Prometheus监控。 工作流程大致分为收集数…

嵌入式Linux之MX6ULL裸机开发学习笔记(汇编LED灯点亮)

汇编LED驱动实验 1.驱动编写 首先创建在vscode上创建工作区&#xff0c;创建led.s汇编文件&#xff0c;然后编写以下程序 .global _start 全局标号 _start: /* 使能所有外设时钟 */ ldr r0,0x020c4068 CCGR0 ldr r1,0xffffffff 要向CCGR0写入的数据 str r1,[r0] 将0xff…

优化企业运营,深入探索SAP库存管理解决方案

SAP库存管理是SAP提供的一款领先的企业库存管理解决方案。它致力于帮助企业实现对库存的全面掌控&#xff0c;优化供应链管理&#xff0c;降低库存成本&#xff0c;提高客户满意度。这个功能强大的系统为企业提供了丰富的仓储管理功能&#xff0c;如库存盘点、物料追踪、供应商…

JAVA批量新增、批量修改

JAVA批量新增、批量修改 若数据量非常大&#xff0c;可以把List拆成多份&#xff0c;每份1000条数据。NetPointDTO批量新增SQL.xml批量修改SQL.xml 若数据量非常大&#xff0c;可以把List拆成多份&#xff0c;每份1000条数据。 import cn.hutool.core.collection.ListUtil; im…

Unity Enum位掩码(BitMask)的运用

Unity Enum位掩码&#xff08;BitMask&#xff09;的运用 前言项目使用场景代码编写定义技能枚举角色类学习技能检查技能 添加并设置脚本运行效果总结 感谢 前言 在Unity游戏开发中&#xff0c;我们经常会面临需要对一组相关的状态进行管理的情况。Enum位掩码是一种有效的方法…

Spring 与 SpringBoot:一窥两者的奥秘与差异

随着 Java 开发领域的不断演进&#xff0c;Spring 框架已经成为了许多企业级应用的首选。然而&#xff0c;近年来&#xff0c;随着 SpringBoot 的兴起&#xff0c;许多开发者开始对其产生了浓厚的兴趣。尽管 SpringBoot 和 Spring 都来自于同一个家族&#xff0c;并且都是为了简…

【LeetCode】150. 逆波兰表达式求值(ASCII码)

今日学习的文章链接和视频链接 leetcode题目地址&#xff1a;150. 逆波兰表达式求值 代码随想录题解地址&#xff1a;代码随想录 题目简介 即将后缀表达式转换成中缀表达式并计算。 给你一个字符串数组 tokens &#xff0c;表示一个根据 逆波兰表示法 表示的算术表达式。 …

【编译原理】期末预习PPT前四章笔记II

看了看学校的ppt&#xff0c;记的比较随意O.o 因为我的考试范围里边没有简答所以概念什么的没怎么记 没有简答只有选择真是太好了嘿嘿嘿 目录 I. 概述&#xff08;好多字。。&#xff09; 一、高级语言的分类 1、体裁 2、执行方式 二、各种语言的执行方式 三、编译程序…

读算法霸权笔记11_微目标

1. 脸书 1.1. 一份请愿书属于脸书了&#xff0c;而社交网络的算法会对如何最大限度地利用这份请愿书做出判断 1.1.1. 脸书的算法在决定谁能看到我的请愿书时会把所有因素都考虑在内 1.2. 通过改变信息推送的方式&#xff0c;脸书研究了我们…

智能分析网关V4智慧港口码头可视化视频智能监管方案

一、需求背景 近年来&#xff0c;水利港口码头正在进行智能化建设&#xff0c;现场管理已经是重中之重。港口作为货物、集装箱堆放及中转机构&#xff0c;具有昼夜不歇、天气多变、环境恶劣等特性&#xff0c;安全保卫工作显得更加重要。港口码头的巡检现场如何高效、快捷地对…

16.Linux Bash Shell通过`read`命令读取用户输入

文章目录 Linux Shell获取用户输入处理简单输入控制等待隐藏输入从文件中读取 欢迎访问个人网络日志&#x1f339;&#x1f339;知行空间&#x1f339;&#x1f339; Linux Shell获取用户输入 处理简单输入 shell与用户交互式输入使用的是read命令&#xff0c;一个简单的例子…

nuxt3 服务端请求其他接口犯的问题与解决

目的 老项目迁移到 nuxt3&#xff0c;为了减少代码修改量&#xff0c;打算封装一个在服务端运行的请求函数&#xff0c;用于在服务端渲染页面的同时&#xff0c;将接口数据请求回来一起返回给客户端 在服务端请求接口并渲染到页面上、在客户端不请求该接口&#xff0c;但需要能…

5G工业物联网网关:连接未来的智能工业

在当今数字化时代&#xff0c;工业物联网正迅速崛起&#xff0c;并引领着全球工业的数字转型。而5G工业物联网网关作为实现IIoT的关键基础设施&#xff0c;在连接未来的智能工业中发挥着举足轻重的作用。 什么是5G工业物联网网关 5G工业物联网网关是连接工业设备和5G网络的关键…

(湖科大教书匠)计算机网络微课堂(下)

第四章、网络层 网络层概述 网络层主要任务是实习网络互连&#xff0c;进而实现数据包在各网络之间的传输 因特网使用TCP/IP协议栈 由于TCP/IP协议栈的网络层使用网际协议IP&#xff0c;是整个协议栈的核心协议&#xff0c;因此TCP/IP协议栈的网络层常称为网际层 网络层提供…

SpringBoot整合sentinel

1、引入依赖 <dependency><groupId>com.alibaba.cloud</groupId><artifactId>spring-cloud-starter-alibaba-sentinel</artifactId> </dependency> 2、 配置文件添加 spring:cloud:sentinel:transport:dashboard: ip:8858 项目重启&#x…

15.三数之和(双指针,C解答附详细分析)

题目描述&#xff1a; 给你一个整数数组 nums &#xff0c;判断是否存在三元组 [nums[i], nums[j], nums[k]] 满足 i ! j、i ! k 且 j ! k &#xff0c;同时还满足 nums[i] nums[j] nums[k] 0 。请 你返回所有和为 0 且不重复的三元组。 注意&#xff1a;答案中不可以包含…