(七)RestAPI 毛子(Http 缓存/乐观锁/Polly/Rate limiting/异步大文件上传)


文章目录

  • 项目地址
  • 一、Http Cache
    • 1.1 服务注册
    • 1.2 Validation with ETag
      • 1. 添加ETagMiddleware中间件
      • 2. 创建内存ETag存储器
      • 3. 服务注册
      • 4. 测试
  • 二、使用ETag实现乐观锁
    • 2.1 添加乐观锁方法
    • 2.2 修改Controller
    • 2.3 测试乐观锁
  • 三、Rate Limiting
    • 3.1 添加速率控制服务
      • 1. 在Program里添加服务
      • 2. 添加AddRateLimiting方法
      • 3. 使用
      • 4. 弊端
  • 四、Idempotent Request 幂等请求
    • 4.1 创建IdempotentRequestAttribute特性
    • 4.2 使用
    • 4.3 测试
  • 五、异步大文件上传
    • 5.1 创建Entities
      • 1. 修改之前的Entry实体
      • 2. EntryImportJob
    • 5.2 创建DTOs
    • 5.3 创建ProcessEntryImportJob


项目地址

  • 教程作者:
  • 教程地址:
  • 代码仓库地址:
  • 所用到的框架和插件:
dbt 
airflow

一、Http Cache

缓存的类型:
1. Client cache
2. Gateway cache(reverse poxy)
3. Proxy cache(CDN)

1.1 服务注册

  1. 注册cache服务
    在这里插入图片描述
  2. 注册中间件

在这里插入图片描述

1.2 Validation with ETag

  • 使用Rest api 返回ETag头部进行缓存
    在这里插入图片描述
    流程:
    1. 第一次请求,没有缓存,数据库查询后,返回并且添加ETag响应头
    2. 响应头存储在浏览器缓存,如果还是相同的请求或者没有更改,则返回304not modified,并从内存缓存里拿数据
    3. 如果更改或者没有缓存数据,则返回新的ETag

1. 添加ETagMiddleware中间件

  • 用于生成Etag头部和判断响应
namespace DevHabit.Api.Middleware;//定义一个中间件,表示请求管道中的下一个中间件
public sealed class ETagMiddleware(RequestDelegate next)
{public async Task InvokeAsync(HttpContext context, InMemoryETagStore eTagStore){//1.如果当前请求方法是 POST、PUT、PATCH 或 DELETE,就跳过 ETag 逻辑if (CanSkipETag(context)){await next(context);return;}//2.获取当前请求的 URI,用作标识资源的 key,稍后要用来生成和比对 ETag。string resourceUri = context.Request.Path.Value!;//3.从请求头中读取客户端带来的 If-None-Match ETag,用于判断资源是否修改过。去掉引号是为了统一格式。string? ifNoneMatch = context.Request.Headers.IfNoneMatch.FirstOrDefault()?.Replace("\"", "");//4.如果请求方法是 GET 或 HEAD,就从 ETag 存储中获取当前资源的 ETagStream originalStream = context.Response.Body; //获取原始响应流using var memoryStream = new MemoryStream(); //创建一个内存流,用于缓存响应内容context.Response.Body = memoryStream; //将响应流写入内存流,以便后续读取响应内容//5.执行请求管道中的下一个中间件(或控制器),并把响应写入 memoryStream 中gawait next(context); //6. 如果响应状态码是 200 OK,并且响应内容类型是 JSON,就计算 ETagif (IsETaggableResponse(context)){memoryStream.Position = 0; //将内存流位置重置到开头byte[] responseBody = await GetResponseBody(memoryStream); //读取内存流中的响应内容string eTag = GenerateETag(responseBody); //计算 ETageTagStore.SetETag(resourceUri, eTag); //将 ETag 存储到 ETag 存储中context.Response.Headers.ETag = $"\"{eTag}\""; //将 ETag 添加到响应头中context.Response.Body = originalStream; //将响应流恢复为原始响应流//9. 如果 ETag 存储中已经有当前资源的 ETag,并且和计算出来的 ETag 一致,就返回 304 Not Modifiedif (context.Request.Method == HttpMethods.Get && ifNoneMatch == eTag) {context.Response.StatusCode = StatusCodes.Status304NotModified;context.Response.ContentLength = 0;return;}}//如果内容有更新,复制缓冲的响应内容到原始响应流中,让客户端收到响应。memoryStream.Position = 0;await memoryStream.CopyToAsync(originalStream);}//判断当前响应是否适合使用 ETag 进行缓存处理private static bool IsETaggableResponse(HttpContext context){return context.Response.StatusCode == StatusCodes.Status200OK &&(context.Response.Headers.ContentType.FirstOrDefault()?.Contains("json", StringComparison.OrdinalIgnoreCase) ?? false);}//读取 MemoryStream 中的响应内容,并以 byte[] 的形式返回private static async Task<byte[]> GetResponseBody(MemoryStream memoryStream){using var reader = new StreamReader(memoryStream, leaveOpen: true);memoryStream.Position = 0;string content = await reader.ReadToEndAsync();return Encoding.UTF8.GetBytes(content);}//根据响应内容生成 ETag 值private static string GenerateETag(byte[] content){byte[] hash = SHA512.HashData(content);return Convert.ToBase64String(hash);}//判断当前请求方法是否可以跳过 ETag 逻辑private static bool CanSkipETag(HttpContext context){return context.Request.Method == HttpMethods.Post ||context.Request.Method 

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

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

相关文章

2025.4.26_STM32_SPI

1.SPI简介 2.硬件电路 所有SPI设备的SCK(时钟)、MOSI(主机输出从机输入)、MISO(主机输入从机输出)分别连在一起。SCK线只能被主机控制&#xff0c;和I2C相同。 主机另外引出多条SS控制线&#xff0c;分别接到各从机的SS引脚 (SS不用的时候为高电平&#xff0c;当主机需要选中某…

JAVA:单例模式

单例模式是设计模式之一 设计模式&#xff0c;就像古代打仗&#xff0c;我们都听过孙子兵法&#xff0c;把计谋概括下来弄成一种模式&#xff0c;形成一种套路。 软件开发中也有很多场景&#xff0c;多数类似的问题场景&#xff0c;解决方案就形成固定的模式&#xff0c;单例…

脑机接口:重塑人类未来的神经增强革命

引言 人类对大脑的探索从未停止&#xff0c;而脑机接口&#xff08;Brain-Computer Interface, BCI&#xff09;的崛起&#xff0c;正在将科幻电影中的“意念操控”变为现实。 这项技术通过解码脑电信号&#xff0c;实现人脑与外部设备的直接交互&#xff0c;不仅为医疗康复带来…

从SOA到微服务:架构演进之路与实践示例

一、架构演进背景 在软件开发领域&#xff0c;架构风格随着业务需求和技术发展不断演进。从早期的单体架构&#xff0c;到面向服务架构(SOA)&#xff0c;再到如今的微服务架构&#xff0c;每一次变革都是为了解决当时面临的核心问题。 二、SOA架构解析 2.1 SOA核心概念 SOA&…

可灵AI 2.0上线:重新定义AI创作?好莱坞级特效触手可及

2025年4月15日&#xff0c;快手正式发布可灵AI 2.0&#xff0c;这款被誉为“让好莱坞特效师颤抖”的AI工具&#xff0c;以物理引擎级动态生成和电影语言自由操控两大核心技术&#xff0c;彻底颠覆了内容创作的想象边界。上线24小时内&#xff0c;全球用户已用它生成超过100万条…

Mysql存储引擎、锁机制

Mysql存储引擎 InnoDB​&#xff08;MySQL 5.5 及以后版本中的默认存储引擎&#xff09; ​​事务支持​​&#xff1a;支持 ​​ACID 事务​​&#xff0c;适合需要高可靠性的场景&#xff08;如支付、订单&#xff09;。 ​​锁机制​​&#xff1a;默认使用 ​​行级锁​​…

飞蛾扑火算法优化+Transformer四模型回归打包(内含MFO-Transformer-LSTM及单独模型)

飞蛾扑火算法优化Transformer四模型回归打包&#xff08;内含MFO-Transformer-LSTM及单独模型&#xff09; 目录 飞蛾扑火算法优化Transformer四模型回归打包&#xff08;内含MFO-Transformer-LSTM及单独模型&#xff09;预测效果基本介绍程序设计参考资料 预测效果 基本介绍 …

音视频开发---视频编码基础

一、视频编码的必要性 1. 存储与传输成本高 未经编码压缩的原始视频的数据量极大,例如:一般电影的亮度信号采样频率为13.5MHz;色度信号的频带通常为亮度信号的一半或更少,为6.75MHz或3.375MHz。以4:2:2的采样频率为例,Y信号采用13.5MHz,色度信号U和V采用6.75MHz采样,…

手动安装OpenSSL1.1.1

根据报错信息 Could not build the ssl module! Python requires a OpenSSL 1.1.1 or newer&#xff0c;说明当前系统中的 OpenSSL 版本低于 1.1.1&#xff0c;或者 Python 编译时未找到符合要求的 OpenSSL 库。以下是分步解决方案&#xff1a; 1. 检查当前 OpenSSL 版本 首先…

[原创](现代Delphi 12指南):[macOS 64bit App开发]: 跨平台开发同样支持retain()引用计数器处理.

[作者] 常用网名: 猪头三 出生日期: 1981.XX.XX 企鹅交流: 643439947 个人网站: 80x86汇编小站 编程生涯: 2001年~至今[共24年] 职业生涯: 22年 开发语言: C/C++、80x86ASM、Object Pascal、Objective-C、C#、R、Python、PHP、Perl、 开发工具: Visual Studio、Delphi、XCode、…

从 “制造” 到 “品牌”:官网建设助力中国企业突围东南亚

在全球产业链重构与区域经济一体化的浪潮下&#xff0c;中国企业出海已从“战略选项”升级为“生存刚需”。东南亚市场因其地理邻近性、人口红利及政策开放性&#xff0c;成为许多中企出海的“首站试验田”。然而&#xff0c;如何在这一文化多元、竞争激烈的市场中建立品牌认知…

iPhone闹钟无法识别调休致用户迟到,苹果客服称会记录反馈

iPhone闹钟无法识别调休致用户迟到&#xff0c;苹果客服称会记录反馈 基于 6 个来源 因“五一”劳动节调休&#xff0c;4月27日&#xff08;周日&#xff09;本应上班&#xff0c;不少iPhone用户却因闹钟未响迟到&#xff0c;“调休”“当苹果闹钟遇到调休”话题登上热搜。苹…

Ubuntu 磁盘空间占用清理(宝塔)

目录 前言1. 基本知识2. 实战 前言 &#x1f91f; 找工作&#xff0c;来万码优才&#xff1a;&#x1f449; #小程序://万码优才/r6rqmzDaXpYkJZF 爬虫神器&#xff0c;无代码爬取&#xff0c;就来&#xff1a;bright.cn 本身自搭建了一个宝塔&#xff0c;突然一下子多了好些空…

杰理-安卓通过map获取时间的时候,部分手机切换sbc和aac时候单耳无声音

杰理-安卓通过map获取时间的时候&#xff0c;部分手机切换sbc和aac时候单耳无声音 #if USER_SUPPORT_PROFILE_MAPif(tws_api_get_role()0){ //主机才获取&#xff0c;否则切换sbc 和 aac 的时候影响单耳无声音user_send_cmd_prepare(USER_CTRL_MAP_READ_TIME,0,NULL);} #endif…

Android 进阶开发:深入掌握 ProgressBar 的使用与高级技巧

一、前言 在 Android 开发中,ProgressBar 是一个非常常见且功能强大的控件,用于向用户反馈操作的进度。虽然它的基础用法简单,但对于进阶开发者来说,如何通过自定义动画、插值器、样式和逻辑控制来提升用户体验,是一个值得深入研究的方向。 本文将带你从 基本使用入手,…

AGILE:开启LLM Agent强化学习的创新框架

在大语言模型&#xff08;LLMs&#xff09;蓬勃发展的今天&#xff0c;基于LLMs构建的智能体成为研究热点。但如何将各组件整合优化仍是难题。本文提出的AGILE框架给出了创新解法&#xff0c;它不仅统一多组件&#xff0c;还让智能体性能超越GPT-4。想知道它是如何做到的吗&…

java使用websocket推送消息到页面

文章目录 一、项目背景二、使用方式1.vue2javaspringpom.xmlRealtimeMonitor.vueMonitorTaskExe.javaWSTopicEnum.javaWServerHelper.java 2.vue3javaspringbootpom.xmlTopologyView.vueAlarmDataInquiryController.javaPushService.javaPushWebSocketHandler.javaWebSocketCon…

小市值策略复现(A股选股框架回测系统)

相关config配置 https://quantkt.com/forumDetail?id201043 很早就知道了小市值模型&#xff0c;正好量化选股回测框架出来了&#xff0c;把最裸的小市值复现下&#xff0c;顺便验证下框架逻辑。 科普: 小市值策略基于 “小市值效应”&#xff0c;即从历史数据来看&#xf…

解决 Flutter 在 iOS 真机上构建失败的问题

在开发 Flutter 应用时&#xff0c;有时会在尝试将应用部署到 iOS 真机时遇到构建失败的问题。错误信息通常类似于以下内容&#xff1a; Could not build the precompiled application for the device. Uncategorized (Xcode): Timed out waiting for all destinations matchi…

OCR(Optical Character Recognition),光学字符识别

参考&#xff1a;如何让机器读懂图片上的文字&#xff1f;飞桨助您快速了解OCR - 知乎 OCR&#xff08;Optical Character Recognition&#xff09;&#xff0c;译为光学字符识别&#xff0c;是指通过扫描等光学输入方式将各种票据、报刊、书籍、文稿及其它印刷品的文字转化为图…