当前位置: 首页 > news >正文

Sentinel核心算法解析の滑动窗口算法

文章目录

  • 前言
  • 一、回顾:快速失败
  • 二、固定窗口算法
  • 三、滑动窗口算法
  • 三、源码体现
    • 3.1、ArrayMetric的初始化
    • 3.2、addPass
    • 3.2.1、currentWindow
    • 3.2.2、wrap.value().addPass
  • 总结


前言

  在Sentinel中,流控效果有快速失败预热排队等待。其中快速失败的统计,利用的就是滑动窗口算法

一、回顾:快速失败

  什么是Sentinel快速失败的效果?例如我有一个接口:
在这里插入图片描述

  进入Sentinel控制台,进行规则设置:
在这里插入图片描述

  如果在1s中,查询次数大于1次,则触发快速失败:
在这里插入图片描述
  当然也可以设置线程数:
在这里插入图片描述

  使用压测工具,设置并发数为5,大于了阈值:
在这里插入图片描述
  同样会触发快速失败:
在这里插入图片描述

二、固定窗口算法

  如果需要实现接口限流,在对于精确度要求不高,并且请求分布较为平均的场景下,常规的计数器法即可满足要求:

  • 在一个固定的时间窗口(比如1分钟)内允许最多 N 个请求(例如 100 个);
  • 记录第一个请求的时间,维护一个计数器;
  • 如果在时间窗口内计数器超过阈值,则拒绝请求;
  • 如果时间窗口过了,重置时间戳和计数器。
public class FixedWindowRateLimiter {private final int limit; // 限流次数private final long intervalMillis; // 时间窗口,单位:毫秒private int counter = 0; // 当前计数器private long windowStart; // 当前窗口开始时间public FixedWindowRateLimiter(int limit, long intervalMillis) {this.limit = limit;this.intervalMillis = intervalMillis;this.windowStart = System.currentTimeMillis();}/*** 尝试请求一次,如果允许返回true,否则返回false*/public synchronized boolean tryRequest() {long now = System.currentTimeMillis();if (now - windowStart >= intervalMillis) {// 超出当前时间窗口,重置计数器和窗口时间counter = 1;windowStart = now;return true;}if (counter < limit) {counter++;return true;} else {// 超过限制,拒绝请求return false;}}public static void main(String[] args) throws InterruptedException {FixedWindowRateLimiter limiter = new FixedWindowRateLimiter(5, 10000); // 10秒内最多5个请求for (int i = 1; i <= 10; i++) {boolean allowed = limiter.tryRequest();System.out.println("请求 " + i + ": " + (allowed ? "通过" : "被拒绝"));Thread.sleep(1000); // 每秒一个请求}}
}

  但是这样的做法是存在弊端的,假如在固定的时间窗口内,请求的情况如下:
在这里插入图片描述
  请求数量不是均匀分布的,虽然0-60s,61-120s这两个区间内的请求数量都没有超过100的阈值,但是在某一个时间段内,比如40s到90s之间,请求的数量超过了阈值(70+60=130)。
  这个问题就是固定时间窗口算法的**“边界问题”**,因为它只在窗口边界上限流,而忽略了滑动区间的请求密度。问题源自于if **(now - windowStart >= intervalMillis) **这一段代码,即只统计了各自窗口边界内的流量。

三、滑动窗口算法

  滑动窗口算法是针对固定窗口算法精度过低的改进。即将请求记录在每个小时间片内(比如每秒),然后滑动计算最近 N 秒内的总请求数,例如同样需要对60s内的请求进行统计和限流,与固定窗口算法不同的是,滑动窗口算法将60s划分成了n等分:
在这里插入图片描述
  每过10s,窗格就会向右边移动一格:
在这里插入图片描述
  每个区间,有着自己独立的计数器,在统计时会用该窗口内最后一个区间的计数 - 该窗口内第一个区间的计数和阈值进行比较。
在这里插入图片描述
  那么这样划分,就能完全避免固定窗口算法存在的边界问题了吗?答案是否定的:假设在75s的时候,来了110个请求,而滑动窗口还在10-70s的区间内(目前的滑动窗口是每隔10s滑动一次),那么75s到15s之间的请求数量一定超过了阈值,但是按照目前的统计情况没有限制住。
在这里插入图片描述
  这次的问题源于粒度。窗口划分的粒度越细,精度也就越高。(上面的案例,如果把精度划成5s,则问题得到了解决)但是相应的对于内存的开销也就越大,并且粒度不可能无限制细化。如果追求严格按时间滑动的统计,可以使用滑动日志算法
  滑动窗口算法的代码简单实现:

  • 每个请求来的时候,把它的时间戳加入列表。
  • 然后遍历 LinkedList,移除那些“窗口之外”的请求记录。
  • 最后看当前列表里的元素个数是否超过限流阈值。
public class SlidingWindowLimiterWithLinkedList {private final int limit; // 窗口内最大请求数private final long windowSizeMillis; // 窗口长度(毫秒)private final LinkedList<Long> requestTimestamps = new LinkedList<>();public SlidingWindowLimiterWithLinkedList(int limit, long windowSizeMillis) {this.limit = limit;this.windowSizeMillis = windowSizeMillis;}public synchronized boolean tryRequest() {long now = System.currentTimeMillis();// 清理过期的请求while (!requestTimestamps.isEmpty() && now - requestTimestamps.peekFirst() > windowSizeMillis) {requestTimestamps.pollFirst(); // 移除最早的请求时间}if (requestTimestamps.size() < limit) {requestTimestamps.addLast(now);return true;} else {return false; // 超过限流阈值}}// 测试代码public static void main(String[] args) throws InterruptedException {SlidingWindowLimiterWithLinkedList limiter = new SlidingWindowLimiterWithLinkedList(5, 10000); // 10秒最多5个请求for (int i = 1; i <= 10; i++) {boolean allowed = limiter.tryRequest();System.out.println("请求 " + i + ": " + (allowed ? "通过" : "被拒绝"));Thread.sleep(1500); // 每1.5秒发一次请求}}
}

三、源码体现

  在StatisticSlot节点的addPassRequest方法中,体现了Sentinel滑动窗口算法的实现。
在这里插入图片描述
  支持分钟级别和秒级别的实现。
在这里插入图片描述

3.1、ArrayMetric的初始化

  ArrayMetric在构造StatisticNode时初始化。
在这里插入图片描述
  LeapArray类型的data属性,实际是OccupiableBucketLeapArray
在这里插入图片描述
在这里插入图片描述
  在其父类LeapArray的构造方法中,对成员变量进行了赋值操作:

  • windowLengthInMs代表了每个窗格的大小,这里是1000 / 2 = 500ms
  • intervalInMs代表了窗格总大小,是1000ms
  • sampleCount代表了窗格个数,是2
  • array是一个存放了时间窗口对象的原子引用数组,默认容量为2(窗格的个数)

在这里插入图片描述
  时间窗口对象WindowWrap< T >是对每个时间窗口的数据包装,泛型T代表窗口里保存的统计数据,比如MetricBucket
  实际是这样的内存结构:

ArrayMetric└── LeapArray< MetricBucket > (字段:leapArray)└── FutureBucketLeapArray (子类实现)└── AtomicReferenceArray<WindowWrap< MetricBucket >>(字段:array)└── WindowWrap< MetricBucket >(每个时间窗口)└── MetricBucket(实际统计数据)

在这里插入图片描述

3.2、addPass

  上面的初始化工作完成后,会进入ArrayMetricaddPass方法,在该方法中做了两件事:

  1. 找到当前时间点所属的那一段时间窗口,内部会根据系统时间计算当前时间落在哪个 WindowWrap 上。
  2. 把这次请求成功的数量加到当前窗口的统计值里,在这个窗口里累加通过的请求数量。

在这里插入图片描述

3.2.1、currentWindow

  在调用currentWindow时,data的实际类型是OccupiableBucketLeapArray。这里传入的是当前时间:
在这里插入图片描述
  currentWindow中的calculateTimeIdx,目的是为了计算当前的时间点会落在哪个时间窗格上
在这里插入图片描述
在这里插入图片描述
  currentWindow中的calculateWindowStart,目的是为了计算当前时间点落在的时间窗格的起始时间
在这里插入图片描述
在这里插入图片描述
  例如传入的参数timeMillis获取到的是800,经过计算得出,会落在第一个时间窗格上,并且该窗格的起点是500:
在这里插入图片描述
  接下来的逻辑都是根据上面计算出的结果,进行一系列的判断,首先从数组中获取计算出下标对应的的WindowWrap对象:
在这里插入图片描述

  • 如果获取到的为空,则会初始化一个,然后通过CAS操作设置回数组对应的下标上(如果存在并发冲突,当前线程会出让时间片,让其他线程去CAS):

在这里插入图片描述

  • 当前时间点落在的时间窗格的起始时间,和从数组中获取计算出下标对应的的WindowWrap对象的起始时间相同,则直接返回该WindowWrap对象。

在这里插入图片描述

  • 当前时间点落在的时间窗格的起始时间,大于从数组中获取计算出下标对应的的WindowWrap对象的起始时间,则会触发更新操作,这也是Sentinel窗口滑动的体现,什么情况会进入这个分支?最典型的,timeMillis为1600,计算出的idx为1,但是windowStart为1500。(idx为1的窗格,时间范围应该是500-1000)

在这里插入图片描述
  选择OccupiableBucketLeapArray的实现:
在这里插入图片描述
  在该方法中,实现了旧窗口计数迁移到新窗口的逻辑:

  • 首先将windowStart属性,设置为calculateWindowStart计算出的当前时间点落在的时间窗格的起始时间

在这里插入图片描述

  • 返回原WindowWrap的MetricBucket (尝试从 borrowArray 中借用历史值)

在这里插入图片描述
- 如果历史值不为空,重置当前窗口的统计对象,然后从 borrowBucket 中把 pass 数 拿过来加到当前窗口。
- 如果历史值为空,只做一个完全的清空。

在这里插入图片描述
  这时的时间窗口:
在这里插入图片描述

  在Sentinel中滑动窗口的更新,体现在数组的每个下标窗格的起始时间和结束时间的更新
  currentWindow方法最后还有一种情况,是针对时钟回拨:
在这里插入图片描述

为什么初始化WindowWrap使用CAS,而更新操作加锁?我想是因为初始化操作耗时短、且只需要成功一次,所以可以用轻量的 CAS。而更新操作逻辑复杂、涉及共享数据的一致性,所以需要加锁来确保线程安全。

3.2.2、wrap.value().addPass

  在currentWindow操作中,拿到了当前时间所在时间窗格的对象WindowWrap后,首先会获取value,这里获取到的是MetricBucket
  MetricBucket中有一个LongAdder[] 原子数组,存放的是当前窗格中请求的数量。
在这里插入图片描述
在这里插入图片描述
  调用addPass,n就是请求个数。
在这里插入图片描述
  MetricEvent是一个枚举类,标识了不同类型的请求:
在这里插入图片描述
  然后调用add方法,给某个类型的指标增加 n 个值。(每个枚举类型,都代表LongAdder[]的一个下标)
在这里插入图片描述

MetricEvent 枚举ordinal 值counters 下标含义
PASS0counters[0]记录通过的请求数
BLOCK1counters[1]记录被拒绝的请求数
SUCCESS2counters[2]记录成功的请求数
EXCEPTION3counters[3]记录发生异常的次数
RT4counters[4]记录响应时间(累计值)

总结

  Sentinel 流控中的快速失败模式,基于其实现的一套滑动窗口算法,支持以 QPS(每秒请求数)或线程数作为控制指标。
  窗口算法分为两种:固定窗口算法滑动窗口算法。其中,固定窗口实现简单,但存在精度较低和边界问题严重的缺陷。滑动窗口是对固定窗口的改进方案:将统计周期进一步细分为多个小窗口,并以固定的时间间隔向前滑动,从而大幅减轻边界问题。窗口越小,精度越高,边界问题越不明显,但也意味着更高的内存开销。
  在 Sentinel 的底层实现中,滑动窗口由 ArrayMetric 负责。其内部维护了一个固定大小为 2 的 WindowWrap 数组(不支持扩容),每个窗口内部包含一个 LongAdder[] 原子数组用于统计数据。该数组中的每个索引对应一个枚举定义的指标项,例如:通过数、阻塞数、异常数、响应时间等。
  Sentinel 在进行实时统计时,主要执行以下两个步骤:

  1. 根据当前时间定位所属的时间窗口;
  2. 对该窗口内的 LongAdder[] 执行累加操作。

  具体流程如下:

  • 首先根据当前系统时间,计算出其所在时间窗格的索引及该窗格的起始时间;

  • 然后从 WindowWrap 数组中获取对应下标的对象;

  • 接着对比当前时间窗口的起始时间与 WindowWrap 对象的起始时间,分为三种情况:

    1. 获取到的窗口为空: 说明该位置尚未初始化,需要创建新窗口;
    2. 起始时间一致: 表示该窗口仍处于有效期内,可直接复用;
    3. 当前窗口起始时间晚于旧窗口: 表示窗口已过期,需要进行重置并替换原有数据。

  以上机制确保了 Sentinel 能够在高并发环境下,以较低开销实现高精度的流量统计与控制。

相关文章:

Sentinel核心算法解析の滑动窗口算法

文章目录 前言一、回顾&#xff1a;快速失败二、固定窗口算法三、滑动窗口算法三、源码体现3.1、ArrayMetric的初始化3.2、addPass3.2.1、currentWindow3.2.2、wrap.value().addPass 总结 前言 在Sentinel中&#xff0c;流控效果有快速失败、预热和排队等待。其中快速失败的统计…...

ida 使用记录

文章目录 伪代码-汇编hexstring快捷键 伪代码-汇编 流程图界面——F5——伪代码界面——再点Tab——流程图界面——再按空格——汇编界面流程图界面——空格——汇编界面 hex view - open subviews - hex dump string view - open subviews - string快捷键&#xff1a; sh…...

数字统计:

1.题意&#xff1a; 在1~N之间寻找d出现的个数&#xff0c;然后输出即可&#xff1b;例如&#xff1a;d2,N23&#xff0c;那么满足条件的有2,12,21,23&#xff0c;所以是4个 2.思路&#xff1a; 1.暴力枚举&#xff08;不可能&#xff09;&#xff1a;可以先写出来去找规律 …...

【架构师从入门到进阶】第五章:DNSCDN网关优化思路——第八节:网关-注入攻击与预防

【架构师从入门到进阶】第五章&#xff1a;DNS&CDN&网关优化思路——第八节&#xff1a;网关-注入攻击与预防 SQL注入攻击的原理攻击者获取数据库表结构预防SQL注入的方法 这篇文章我们来看SQL注入。 SQL注入攻击的原理 SQL注入攻击的原理呢&#xff1f;我们来简单说…...

波束形成(BF)从算法仿真到工程源码实现-第五节-线性约束最小方差波束形成算法(LCMV)

一、概述 本节我们讨论线性约束最小方差波束形成算法(Linearly constrained minimum variance,LCMV)波束形成算法&#xff0c;包括原理分析及代码实现。 更多资料和代码可以进入https://t.zsxq.com/qgmoN &#xff0c;同时欢迎大家提出宝贵的建议&#xff0c;以共同探讨学习。 …...

Java类加载机制原理与应用

前言 Java 中的类加载机制&#xff08;Class Loading Mechanism&#xff09;是 JVM 架构中的核心组成部分&#xff0c;它控制着类从编译后的 .class 文件被加载到内存、并最终变成可以被程序使用的对象的全过程。涉及类加载器、双亲委派模型及加载过程。下面我们从原理到实际应…...

android display 笔记(十三)surfcaeflinger的DEQUEUED、QUEUED

BufferQueue 的核心作用 BufferQueue 是 生产者-消费者模型 的核心组件&#xff0c;协调应用&#xff08;生产者&#xff09;和 SurfaceFlinger&#xff08;消费者&#xff09;之间的图形缓冲区&#xff08;GraphicBuffer&#xff09;传递。 生产者&#xff1a;应用&#xff0…...

数据库预热

介绍 Database Warm-up &#x1f9e0; 一句话理解 数据库是在应用启动阶段&#xff0c;提前建立数据库连接 或 执行轻量 SQL 操作&#xff0c;从而 加快首个请求的响应速度 的一种优化手段 &#x1f3af; 为什么需要数据库预热&#xff1f; 当 FastAPI 或其他 Web 服务刚启…...

C语言—程序的编译和链接

1. 翻译环境和运行环境 在ANSI S的任何一种实现中&#xff0c;存在两个不同的环境 第一种是翻译环境&#xff0c;在这个环境中源代码被转换为可执行的机器指令&#xff08;二进制指令&#xff09; 第二种是执行环境&#xff0c;它用于实际执行代码 2. 翻译环境 翻译环境是由…...

Neo4j GDS-10-neo4j GDS 库中相似度算法介绍

neo4j apoc 系列 Neo4j APOC-01-图数据库 apoc 插件介绍 Neo4j GDS-01-graph-data-science 图数据科学插件库概览 Neo4j GDS-02-graph-data-science 插件库安装实战笔记 Neo4j GDS-03-graph-data-science 简单聊一聊图数据科学插件库 Neo4j GDS-04-图的中心性分析介绍 Ne…...

Unity 动画

Apply Root Motion 勾选的话就会使用动画片段自带的位移 Update Mode &#xff08;动画重新计算骨骼位置转向缩放的数值&#xff09;&#xff1a; Normal &#xff1a; 随Update走&#xff0c;每次Update都计算Animate Physics &#xff1a;与 fixed Update() 同步&#xff0…...

【位运算】只出现一次的数字 II

文章目录 137. 只出现一次的数字 II解题思路一&#xff1a;借用数组的位运算解法二&#xff1a;不使用数组的位运算 137. 只出现一次的数字 II 137. 只出现一次的数字 II ​ 给你一个整数数组 nums &#xff0c;除某个元素仅出现 一次 外&#xff0c;其余每个元素都恰出现 **…...

模型开发中的微调是干什么

在模型开发中&#xff0c;微调&#xff08;Fine-tuning&#xff09; 是指利用预训练模型&#xff08;Pre-trained Model&#xff09;的参数作为初始值&#xff0c;在特定任务或数据集上进一步调整模型参数的过程。它是迁移学习&#xff08;Transfer Learning&#xff09;的核心…...

leetcode 204. Count Primes

题目描述 这是道纯数学类问题。 先回忆一下&#xff0c;素数的定义。 质数&#xff08;英文名&#xff1a;Prime number&#xff09;又称素数&#xff0c;是指在大于1的自然数中&#xff0c;除了1和它本身以外不再有其他因数的自然数。 质数又称素数。一个大于1的自然数&…...

fastadmin后端添加页面,自主控制弹出框关闭,关闭父页面弹框

Form.api.bindevent($(“form[roleform]”), (data, ret) > { 重写绑定事件,返回false即可 注意:只有返回code1才能拦截,其他值不进行拦截 add: function () {//获取当前search里面的type值var type location.search.split(type)[1];Form.api.bindevent($("form[role…...

LeetCode 255 超通俗讲解:Swift 验证前序是否 BST

文章目录 摘要描述题解答案题解代码分析核心点解释&#xff1a; 示例测试及结果时间复杂度空间复杂度总结未来展望 摘要 在做算法题的时候&#xff0c;树相关的题总是“神神叨叨”的&#xff0c;但其实抓住核心规则&#xff0c;它们也挺有逻辑的。今天这题——LeetCode 255&am…...

Win32++ 使用初探

文章目录 1. 环境要求2. Win32安装3. 项目创建3.1 项目创建&#xff08;1&#xff09;直接使用Win32里的示例Sample&#xff08;2&#xff09;自行创建项目 最近想用 VC写些 UI&#xff0c;但又不太想用 MFC&#xff0c;正好对界面要求不太高&#xff0c;就使用了一下 Win3…...

求解时间复杂度

1.设 t 法 当求解出现while循环时&#xff0c;设t求解 void fun(int n) {int i 1;while(i < n)i i * 2; } 解法&#xff1a; 1.设循环次数为t&#xff1b; 2.将while循环中的语句展开到循环t次 1 2 3 …… t 2 2^2 2^3 …… 2^t 3.跳出循环 2^t > n …...

深度解析:如何高效识别并定位问题关键词

什么是问题关键词&#xff1f; 问题关键词是人们在搜索引擎中输入以查找信息、答案或解决方案的问题。这些查询以问题指示符开头&#xff0c;例如&#xff1a; who、what、where、when、why、how、which、will、would、should、can、could、is、are、was、were、do、does 或 d…...

c++小做——完全数

今天&#xff0c;我们来写一个完全数的代码 首先是 long long n; cin>>n; &#xff08;you~输入的数&#xff09; 然后是 long long b0;//因数的和 long long cnt0;//计数器 接着是 for(long long i2;i<n-1;i) {} 在里面插入 bb-i;再写一个for for(int a1;a&…...

GGML源码逐行调试(下)

目录 前言1. 简述2. 预分配计算图内存2.1 创建图内存分配器2.2 构建最坏情况的计算图2.3 预留计算图内存 3. 分词4. 模型推理与生成4.1 模型推理4.2 采样 结语下载链接参考 前言 学习 UP 主 比飞鸟贵重的多_HKL 的 GGML源码逐行调试 视频&#xff0c;记录下个人学习笔记&#x…...

JavaScript学习教程,从入门到精通, JavaScript 函数全面解析与案例实践(11)

JavaScript 函数全面解析与案例实践 项目导读 JavaScript 函数是编程中的核心概念&#xff0c;是执行特定任务的代码块。本教程将全面讲解函数的定义、参数、返回值及调用方式&#xff0c;并通过实际案例加深理解。 学习目标 掌握 JavaScript 函数的定义与调用方法理解函数…...

音视频之H.265/HEVC编码框架及编码视频格式

一、编码框架&#xff1a; H.265/HEVC采用混合编码框架&#xff0c;包括变换、量化、熵编码、帧内预测、帧预测以及环路滤波等模块。但是&#xff0c;H.265/HEVC几乎在每个模块都引入了新的编码技术。 1、帧内预测&#xff1a; 该模块主要用于去除图像的空间相关性。通过编码后…...

栈与队列:两种经典线性数据结构的深度解析

一、栈&#xff1a;LIFO 特性的完美诠释 &#xff08;一&#xff09;核心概念与抽象模型 定义与特性 栈是一种严格遵循后进先出&#xff08;LIFO&#xff09;原则的线性数据结构&#xff0c;其操作被限制在栈顶&#xff08;Top&#xff09;进行。形象化理解&#xff1a;如同堆…...

0x01、Redis 主从复制的实现原理是什么?

Redis 主从复制概述 Redis 的主从复制是一种机制&#xff0c;允许一个主节点&#xff08;主实例&#xff09;将数据复制到一个或多个从节点&#xff08;从实例&#xff09;。通过这一机制&#xff0c;从节点可以获取主节点的数据并与之保持同步。 复制流程 开始同步&#xf…...

Python实现贪吃蛇一

贪吃蛇是一款经典的小游戏&#xff0c;最近尝试用Python实现它。先做一个基础版本实现以下目标&#xff1a; 1、做一个按钮&#xff0c;控制游戏开始 2、按Q键退出游戏 3、右上角显示一个记分牌 4、随机生成一个食物&#xff0c;蛇吃到食物后长度加一&#xff0c;得10分 5、蛇碰…...

01-libVLC的视频播放器:环境搭建以及介绍

项目展示项目播放器 VLC简介VLC媒体播放器(VideoLAN Client)是一款开源、跨平台的自由多媒体播放器,由VideoLAN项目开发。它支持众多音频与视频格式(如MPEG-2、MPEG-4、H.264、MKV、WebM、WMV、MP3等),以及DVD、VCD和各种流媒体协议。 VLC的特点跨平台支持:Windows、mac…...

linux内核升级

这里介绍一下linux内核升级 因为需要搭建k8s集群内核内核版本过低会导致集群出现问题&#xff0c;为了避免问题发生我们对集群内核进行升级处理 这个是我目前本身的内核版本 用了很多的镜像站去进行更新发现更新不了&#xff08;阿里云不能用了&#xff0c;貌似是删除了&…...

电感详解:定义、作用、分类与使用要点

一、电感的基本定义 电感&#xff08;Inductor&#xff09; 是由导线绕制而成的储能元件&#xff0c;其核心特性是阻碍电流变化&#xff0c;将电能转化为磁能存储。 基本公式&#xff1a; 自感电动势&#xff1a; E -L * (di/dt) &#xff08;L&#xff1a;电感值&#xff0c…...

扩散模型简介

扩散模型简介 基本原理 扩散模型是一种基于概率扩散过程的生成模型&#xff0c;其核心思想是通过正向扩散过程和反向去噪过程生成数据&#xff1a; 正向扩散过程&#xff1a;从真实数据&#xff08;如图像&#xff09;开始&#xff0c;逐步添加高斯噪声&#xff0c;最终将数据…...

MySQL安装实战分享

一、在 Windows 上安装 MySQL 1. 下载 MySQL 安装包 访问 MySQL 官方下载页面。选择适合你操作系统的版本。一般推荐下载 MySQL Installer。 2. 运行安装程序 双击下载的安装文件&#xff08;例如 mysql-installer-community-<version>.msi&#xff09;。如果出现安全…...

掌握 Git 的十大基础命令

李升伟 编译 在 IT 领域&#xff0c;很少有技术能像 Git 一样占据绝对主导地位&#xff0c;几乎无人能及。Git 在软件开发中扮演着核心角色&#xff0c;其影响力之大甚至让其他版本控制系统&#xff08;如 SVN 和 Mercurial&#xff09;几乎被淘汰。如今&#xff0c;我们已难以…...

58-使用wordpress快速创建个人网站

直接找台可以联网的linux&#xff08;我的环境是rocky8.9&#xff09;一顿运行&#xff0c;思路就是安装docker&#xff0c;然后启动一个数据库&#xff0c;然后启动一个wordpress&#xff0c;然后就是把端口暴露出来。 227 yum remove podman 228 yum install -y yum-utils…...

若依前后端分离版运行教程、打包教程、部署教程

后端打包教程 注意&#xff1a;需要先运行redis 2、前端运行教程 2.1安装依赖 2.2运行 打开浏览器查看,地址&#xff1a;http://localhost:80 3、前端打包教程 3.1打包 3.2运行打包好的文件&#xff0c;先找到打包好的文件 这是nginx的文件结构 将打包好的文件放到html目录下…...

【Python3教程】Python3基础篇之数据结构

博主介绍:✌全网粉丝22W+,CSDN博客专家、Java领域优质创作者,掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域✌ 技术范围:SpringBoot、SpringCloud、Vue、SSM、HTML、Nodejs、Python、MySQL、PostgreSQL、大数据、物联网、机器学习等设计与开发。 感兴趣的可…...

transformers的 pipeline是什么:将模型加载、数据预处理、推理等步骤进行了封装

transformers的 pipeline是什么:将模型加载、数据预处理、推理等步骤进行了封装 pipe = pipeline("text-generation", model=model, tokenizer=tokenizer, max_new_tokens=50 )pipeline :这是 transformers 库中一个非常实用的工具函数。它可以基于预训练模型快速构…...

十七、TCP编程

TCP 编程是网络通信的核心&#xff0c;其 API 围绕面向连接的特性设计&#xff0c;涵盖服务端和客户端的交互流程。以下是基于 ​C 语言的 TCP 编程核心 API 及使用流程的详细解析&#xff1a; 核心 API 概览 ​函数​角色​描述socket()通用创建套接字&#xff0c;指定协议族…...

Obsidian 技巧篇

Obsidian 技巧篇 本篇文章主要汇总分享几个 Ob 中好用的小技巧&#xff0c;包括嵌入视频播放、文本颜色设置、插入大纲、Mermaid 绘制图形。原文见于&#xff1a;Obsidian技巧篇。 嵌入视频播放 <iframe width"860" height"700" src"https://ww…...

使用Fortran读取HDF5数据

使用Fortran读取HDF5数据 下面我将介绍如何在Fortran中读取HDF5文件中的各种类型数组数据&#xff0c;包括一维数组、二维数组、元数组和变长数组。 准备工作 首先需要确保系统安装了HDF5库&#xff0c;并且在编译时链接了HDF5库。例如使用gfortran编译时&#xff1a; gfor…...

L36.【LeetCode题解】查找总价格为目标值的两个商品(剑指offer:和为s的两个数字) (双指针思想,内含详细的优化过程)

目录 1.LeetCode题目 2.分析 方法1:暴力枚举(未优化的双指针) 方法2:双指针优化:利用有序数组的单调性 版本1代码 提问:版本1代码有可以优化的空间吗? 版本2代码 提问:版本2代码有可以优化的空间吗? 版本3代码(★推荐★) 3.牛客网题目:和为s的数字 1.LeetCode题目 …...

mysql 商城商品属性开发的动态解决方案

终极方案&#xff1a;动态属性解决方案 推荐使用 JSON 字段 虚拟列索引 的组合方案 结合灵活存储与查询优化&#xff0c;平衡扩展性与性能 完整实现步骤 步骤 1&#xff1a;创建基础表结构 CREATE TABLE products (id INT PRIMARY KEY AUTO_INCREMENT,name VARCHAR(100) NO…...

Java递归练习----猴子偷桃

问题&#xff1a; 有一堆桃子&#xff0c;猴子第一天吃灵其中的一般&#xff0c;并在多吃了一个&#xff01;以后每天猴子都吃其中的一半&#xff0c;然后多吃一个。当到第十天时&#xff0c;想再吃时&#xff08;即还没吃&#xff09;&#xff0c;发现只有1个桃子了&#xff…...

[干货]PHM学习软件|PHM预测性维护系统

使用步骤教程如下 1、登录 用户名&#xff1a;52phm 密码&#xff1a;xxx &#xff08;区别在于不同用户密钥不一样&#xff09; 2、上传需要分析的数据集 支持数据集格式&#xff1a;csv、xlsx、xls、mat、json 3、主题1&#xff1a;机械参数计算 计算轴承、齿轮、皮带的…...

详解正则表达式中的?:、?= 、 ?! 、?<=、?<!

1、?: - 非捕获组 语法: (?:pattern) 作用: 创建一个分组但不捕获匹配结果&#xff0c;不会将匹配的文本存储到内存中供后续使用。 优势: 提高性能和效率 不占用编号&#xff08;不会影响后续捕获组的编号&#xff09; 减少内存使用 // 使用捕获组 let regex1 /(hell…...

Java常见面试问题

一.Liunx 二.Java基础 1.final 2.static 3.与equals 三.Collection 1.LIst 2.Map 3.Stream 四、多线程 1.实现方法 2.线程池核心参数 3.应用场景 五、JVM 1.堆 2.栈 六、Spring 1.面向对象 2.IOC 3.AOP 七、Springboot 1.自动装配 八、SpringCloud 1.Nacos 2.seata 3.ga…...

C#MQTT协议服务器与客户端通讯实现(客户端包含断开重连模块)

C#MQTT协议服务器与客户端通讯实现 1 DLL版本2 服务器3 客户端 1 DLL版本 MQTTnet.DLL版本-2.7.5.0 基于比较老的项目中应用的DLL&#xff0c;其他更高版本变化可能较大&#xff0c;谨慎参考。 2 服务器 开启服务器 关闭服务器 绑定事件【客户端连接服务器事件】 绑定事件【客户…...

GGML源码逐行调试(上)

目录 前言1. 简述2. 环境配置3. ggml核心概念3.1 gguf3.2 ggml_tensor3.3 ggml_backend_buffer3.4 ggml_context3.5 backend3.6 ggml_cgraph3.7 ggml_gallocr 4. 推理流程整体梳理4.1 时间初始化与参数设置4.2 模型加载与词汇表构建4.3 计算图与内存分配4.4 文本预处理与推理过…...

智能测试用例生成:老旧平台页面查询功能的大模型改造

引言 由于GUI小工具【Deepseek APIPython 测试用例一键生成与导出】的不断升级实践&#xff0c;发现大模型的需求文档解析生成测试用例的可直接复用率不太理想&#xff0c;因此萌生了对老旧系统升级改造的想法。旧测试用例生成平台主要在于采集用户输入的字段名称、字段类型及…...

使用Python解决Logistic方程

引言 在数学和计算机科学中,Logistic 方程是描述人口增长、传播过程等现象的一种常见模型。它通常用于表示一种有限资源下的增长过程,比如动物种群、疾病传播等。本文将带领大家通过 Python 实现 Logistic 方程的求解,帮助你更好地理解这一经典数学模型。 1.什么是 Logist…...

文件上传基本原理靶场实现

一. 漏洞原理 未经验证的上传机制&#xff1a; 应用程序未对用户上传的文件进行充分验证&#xff0c;包括&#xff1a; 文件类型/扩展名&#xff1a;仅依赖客户端提交的MIME类型或简单检查扩展名&#xff08;如.jpg&#xff09;&#xff0c;但未验证文件实际内容。 文件内容&a…...