深入浅出限流算法(二):更平滑的滑动窗口
好的,接续上一篇关于固定窗口计数器的讨论,我们现在来看看它的改进版——滑动窗口算法,它旨在解决固定窗口那个恼人的“临界突变”问题。
在上一篇文章中,我们探讨了最简单的固定窗口计数器限流算法,并指出了它最大的缺陷——临界突变问题,即在窗口边界可能产生远超预期的瞬时流量。为了克服这个问题,工程师们设计出了更精妙的滑动窗口 (Sliding Window) 算法。
滑动窗口的核心思想
想象一下,你不是每分钟换一个全新的计数器,而是有一个总时长固定(比如 1 分钟)但由多个小格子组成(比如 6 个 10 秒的格子)的观察窗口。
- 细分窗口: 时间被划分成连续的小格子(子窗口),每个格子记录自己时间段内的请求数。
- 滑动观察: 你的观察窗口始终覆盖着最近的若干个格子(总时长固定,例如始终覆盖最近 6 个 10 秒的格子,构成一个 1 分钟的滑动窗口)。
- 计算总和: 当新请求到来时,你计算当前观察窗口内所有格子计数器的总和。
- 检查限制: 如果总和小于等于阈值(例如 QPS 限制),则允许请求,并将当前请求计入其所属的那个小格子。如果超过阈值,则拒绝请求。
- 时间推移: 随着时间的流逝,观察窗口整体向前“滑动”:最老的一个格子(及其计数)滑出观察窗口,不再参与计算;最新的一个时间格子进入观察窗口。
通过这种方式,统计总是基于一个持续滑动的、固定长度的时间区间,避免了固定窗口在边界处的计数器突然清零,从而使得流量限制更加平滑。
Java 实现
下面是一个基于数组的滑动窗口限流器 Java 实现:
import java.time.LocalTime; // 这个 import 在代码中实际未使用,可以移除
import java.util.concurrent.atomic.AtomicInteger;public class RateLimiterSlidingWindow {private int qps = 2; // 总的 QPS 限制 (每秒允许请求数)private long windowSize = 1000; // 主窗口大小 (毫秒),例如 1000ms = 1秒private int windowCount = 10; // 子窗口(格子)的数量private WindowInfo[] windowArray; // 存储子窗口数据的数组// 内部类,用于存储每个子窗口的信息private static class WindowInfo {private Long time; // 此子窗口最后一次被重置或更新的时间戳private AtomicInteger number; // 此子窗口内的请求计数器 (线程安全)// 构造函数public WindowInfo(long time, AtomicInteger number) {this.time = time;this.number = number;}// Getters and Setters (可以根据需要添加)public Long getTime() { return time; }public void setTime(Long time) { this.time = time; }public AtomicInteger getNumber() { return number; }}// 构造函数:初始化限流器public RateLimiterSlidingWindow(int qps, long windowSize, int windowCount) {this.qps = qps;this.windowSize = windowSize;this.windowCount = windowCount;this.windowArray = new WindowInfo[this.windowCount];long currentTimeMillis = System.currentTimeMillis();// 初始化所有子窗口,初始时间设为当前,计数设为0for (int i = 0; i < windowArray.length; i++) {windowArray[i] = new WindowInfo(currentTimeMillis, new AtomicInteger(0));}}// 尝试获取一个请求许可 (核心方法)// 使用 synchronized 保证对 windowArray 状态修改的线程安全public synchronized boolean tryAcquire() {long currentTimeMillis = System.currentTimeMillis();// 1. 计算当前时间戳应该落在哪个子窗口 (格子)long subWindowSize = windowSize / windowCount; // 每个子窗口的时间跨度// 通过取模和除法,将当前时间映射到数组索引int currentIndex = (int)((currentTimeMillis % windowSize) / subWindowSize);// 2. 遍历所有子窗口:重置过期窗口 & 计算当前滑动窗口总数int sum = 0; // 当前滑动窗口内的总请求数for (int i = 0; i < windowArray.length; i++) {WindowInfo windowInfo = windowArray[i];// 检查子窗口是否已过期 (时间戳距离现在超过整个主窗口大小)if ((currentTimeMillis - windowInfo.getTime()) > windowSize) {// 如果过期,重置计数器为 0,并更新其时间戳为当前时间// 这相当于让这个格子“重新进入”滑动窗口的循环利用windowInfo.getNumber().set(0);windowInfo.setTime(currentTimeMillis);}// 如果当前遍历到的格子就是本次请求所属的格子if (currentIndex == i) {// 先尝试将当前格子的计数加 1 (后面如果超限会撤销)windowInfo.getNumber().incrementAndGet();}// 累加所有未过期格子的计数sum = sum + windowInfo.getNumber().get();}// 3. 检查总数是否超过限制if (sum > qps) {// 如果总数超限,说明刚才对 currentIndex 的增加是无效的// 需要撤销 (回滚) 这个增加操作windowArray[currentIndex].getNumber().decrementAndGet();return false; // 拒绝请求} else {// 如果未超限,刚才的增加有效,请求被允许return true;}}// main 方法用于测试 (可以参考上一篇的测试方式)public static void main(String[] args) throws InterruptedException {// QPS=2, 窗口1秒, 分成10个格子 (每个100ms)RateLimiterSlidingWindow limiter = new RateLimiterSlidingWindow(2, 1000, 10);for (int i = 0; i < 10; i++) {// 模拟请求间隔,观察效果Thread.sleep(100); // 每 100ms 发一个请求// Thread.sleep(400); // 每 400ms 发一个请求// Thread.sleep(600); // 每 600ms 发一个请求long currentTimeMillis = System.currentTimeMillis();if (limiter.tryAcquire()) {System.out.println(currentTimeMillis + " 请求成功 -> ✅");} else {System.out.println(currentTimeMillis + " 请求失败 -> ❌");}}}
}
代码解读:
-
WindowInfo 内部类: 存储每个子窗口(格子)的时间戳和原子计数器。
-
windowArray: 一个 WindowInfo 对象的数组,数组大小即格子的数量 (windowCount)。
-
构造函数: 初始化 qps、windowSize、windowCount,并创建和初始化 windowArray。所有格子的初始时间戳设为当前时间,计数为 0。
-
tryAcquire() 核心逻辑:
-
计算 currentIndex: 确定当前请求应该落在哪一个格子里。通过 (当前时间 % 主窗口大小) / 子窗口大小 的方式,巧妙地将无限的时间轴映射到有限的数组索引上,实现了循环使用数组格子的效果。
-
遍历 windowArray:
- 重置过期格子: 对每个格子,检查其记录的时间戳 windowInfo.getTime() 是否已经“太旧”(距离当前时间超过了一个 windowSize)。如果是,说明这个格子代表的时间段已经完全滑出了当前的统计窗口,需要将其计数清零,并更新时间戳为当前时间,为后续循环利用做准备。
- 增加当前格子计数: 如果遍历到的格子 i 正好是当前请求所属的格子 currentIndex,就先尝试将其计数原子地加 1。
- 累加总数: 将所有(未过期的)格子的计数值累加到 sum 中。
-
检查并处理结果:
- 如果 sum 超过了 qps 限制,说明刚才对 currentIndex 的增加导致了超限,需要调用 decrementAndGet() 将其撤销,并返回 false (拒绝)。
- 如果 sum 未超过 qps,说明请求被允许,刚才的增加有效,返回 true。
-
-
线程安全: 同样使用了 synchronized 关键字来保证对共享数组 windowArray 的访问和修改是线程安全的。AtomicInteger 保证了单个格子计数的原子性。
优点
- 缓解临界突变:通过细化时间窗口并进行滑动统计,显著减少了固定窗口在边界处可能出现的流量突刺问题,使得流量控制更加平滑。
- 实现相对简单:相比更复杂的算法(如令牌桶),滑动窗口的概念和基于数组的实现仍然比较直观。
缺点与考量
- 并未完全消除突变:滑动窗口虽然平滑了流量,但仍然可能存在一定的突发性。例如,如果子窗口(格子)划分得不够细,在子窗口的边界处仍然可能出现小范围的流量聚集。精度取决于格子的数量,格子越多越平滑,但内存和计算开销也越大。
- synchronized 性能瓶颈:在高并发场景下,synchronized 可能会成为性能瓶颈,因为它锁定了整个方法,限制了并发度。更优化的实现可能会采用更细粒度的锁或者无锁数据结构(但这会显著增加实现的复杂度)。
- 内存占用:相比固定窗口,需要存储多个子窗口的信息,内存占用有所增加。
总结
滑动窗口算法是对固定窗口计数器算法的有效改进,通过将时间窗口细分为多个格子并进行滑动统计,显著缓解了临界突变问题,提供了更平滑、更准确的流量控制。虽然它并非完美(仍有一定突发可能,且简单实现存在并发瓶颈),但它在精度和实现复杂度之间取得了较好的平衡,是许多限流场景下的常用选择。
相关文章:
深入浅出限流算法(二):更平滑的滑动窗口
好的,接续上一篇关于固定窗口计数器的讨论,我们现在来看看它的改进版——滑动窗口算法,它旨在解决固定窗口那个恼人的“临界突变”问题。 在上一篇文章中,我们探讨了最简单的固定窗口计数器限流算法,并指出了它最大的缺…...
纷析云开源财务软件:基于Spring Boot的轻量化财务中台实践
一、技术架构与核心设计 全栈开源技术栈 后端框架:基于Spring Boot 3.x构建,集成MyBatis-Plus作为ORM层,支持JDK 17特性(如虚拟线程并发处理),确保高吞吐与稳定性。 前端框架:采用Vue 3 TypeS…...
软考-软件设计师中级备考 5、数据结构 树和二叉树
1、树的基本概念 节点的度:节点拥有的子树数目。例如,若一个节点有 3 棵子树,其度为 3。树的度:树中节点的最大度数。如树中所有节点的度最大为 4,则树的度是 4。叶子节点:度为 0 的节点,也…...
php 需要学会哪些技术栈,掌握哪些框架
作为一个「野生」程序员,我的学习过程比较急功近利。 我记得自己写的第一个 PHP 程序是留言本。一上来对 PHP 一窍不通,所以直接去网上找了个留言本的源码,下载下来后先想办法让它在自己电脑上运行起来。通过这个过程掌握了 PHP 开发环境的搭…...
短视频矩阵系统贴牌批量剪辑功能开发,支持OEM
一、引言 在短视频行业蓬勃发展的当下,短视频矩阵运营已成为企业和个人实现品牌推广、流量增长的重要策略。然而,面对大量的视频素材和多个运营账号,传统的单个视频剪辑、发布方式效率极低,难以满足矩阵运营的需求。为了提高内容…...
【Java EE初阶】多线程(二)
1.在图中代码,我们调用了start方法,真正让系统调用api创建了一个新线程,而在这个线程跑起来之后,就会自动执行到run。调用start方法动作本身速度非常快,一旦执行,代码就会立即往下走,不会产生任…...
分布式链路追踪理论
基本概念 分布式调用链标准-openTracing Span-节点组成跟踪树结构 有一些特定的变量,SpanName SpanId traceId spanParentId Trace(追踪):代表一个完整的请求流程(如用户下单),由多个Span组成…...
conda和bash主环境的清理
好的!要管理和清理 Conda(或 Bash)安装的包,可以按照以下步骤进行,避免冗余依赖,节省磁盘空间。 📌 1. 查看已安装的包 先列出当前环境的所有安装包,找出哪些可能需要清理ÿ…...
Linux系统管理与编程14:Shell变量及定制bash登录界面
兰生幽谷,不为莫服而不芳; 君子行义,不为莫知而止休。 1.准备工作 创建用户wu useradd wu passwd wu 修改权限 chmod uw /etc/sudoers 编辑 visudo 在root行下,添加:“wu ALL……” 图14- 1 恢复文件权限并…...
微信小程序开发笔记
一、首先,下载一个微信开发者工具。前端项目就正常创建,由于本人的前端一塌糊涂,就让AI给我生成了一个我想要的前端项目(包括后面写功能)。 这里开发的时候会用到这个,但是一定注意服务部署到服务器上再本…...
SEO长尾关键词优化核心策略
内容概要 在搜索引擎优化领域,长尾关键词因其精准的流量捕获能力与较低的竞争强度,已成为提升网站自然流量的核心突破口。本文围绕长尾关键词优化的全链路逻辑,系统拆解从需求洞察到落地执行的五大策略模块,涵盖用户搜索意图解析…...
第一节:Linux系统简介
理论知识 Linux的起源与发展:1991 年,芬兰赫尔辛基大学的学生林纳斯托瓦兹受到 Minix 和 Unix 思想的启发,开始编写 Linux 内核。最初,它只是一个个人项目,但随着开源社区的加入,Linux 迅速发展壮大。如今…...
微信聊天机器人搭建 教程/开发
创建标签 简要描述: 添加标签 请求URL: http://域名地址/addContactLabel 请求方式: POST 请求头Headers: Content-Type:application/jsonAuthorization:login接口返回 参数: 参数名必…...
Ubuntu中C++项目安装二次规划库——qpOASES 库
一、在Ubuntu安装qpOASES 库 步骤 1:更新系统包列表 首先,打开终端,执行以下命令更新系统的包列表,以确保你能获取到最新的软件包信息。 sudo apt update 步骤 2:安装必要的依赖 qpOASES库的编译和安装需要一些基…...
JavaScript-基础语法
前言: 一个网页由三个部分组成: 1.html:超文本标记语言,用于控制网页的结构(页面元素和内容) 2.css:级联样式表,用于控制网页布局,涉及对网页文字,背景,布局进…...
已有 npm 项目,如何下载依赖、编译并运行项目
诸神缄默不语-个人技术博文与视频目录 这篇博文的适用场景是比如说反正你现在有了一个现成的npm项目,然后无论如何,你要把前端挂起来。 文章目录 一、准备工作1. 安装 Node.js 和 npm2. 克隆或获取项目代码 二、安装项目依赖三、了解 npm 脚本命令四、构…...
第四章:Messaging and Memory
Chapter 4: Messaging and Memory 从配置管理到消息记忆:如何让AI记住对话内容? 在上一章的配置管理中,我们已经能让系统记住所有参数设置。但你是否想过:如果用户连续提问“今天天气如何?”和“明天呢?”…...
iPhone闹钟无法识别调休致用户迟到,苹果客服称会记录反馈
iPhone闹钟无法识别调休致用户迟到,苹果客服称会记录反馈 基于 6 个来源 因“五一”劳动节调休,4月27日(周日)本应上班,不少iPhone用户却因闹钟未响迟到,“调休”“当苹果闹钟遇到调休”话题登上热搜。苹…...
npm error code CERT_HAS_EXPIRED
npm error code CERT_HAS_EXPIRED 欢迎来到我的主页,我是博主英杰,211科班出身,就职于医疗科技公司,热衷分享知识,武汉城市开发者社区主理人 擅长.net、C、python开发, 如果遇到技术问题,即可私…...
C++ 之 【list的简介、list 的构造函数、iterator、容量操作、元素访问、增删查改与迭代器失效】
目录 1.list的介绍 2.list的使用 2.1 构造函数 2.2 iterator 的使用 2.3 容量操作 2.4 元素访问 2.5 增删查改 2.5.1头插头删与尾插尾删 2.5.2 insert 、erase 函数 2.5.3 clear、swap函数 2.5.4 关于find函数 3.迭代器失效 1.list的介绍 (1)list的底层通常实现为带…...
使用手机录制rosbag包
文章目录 简介录制工具录制步骤录制设置设置IMU录制频率设置相机分辨率拍照模式录制模式数据制作获取数据数据转为rosbag查看rosbag简介 ROS数据包(rosbag)是ROS系统中用于记录和回放传感器数据的重要工具,通常用于算法调试、系统测试和数据采集。传统上,rosbag依赖于ROS环…...
使用阿里云 CDN 保护网站真实 IP:完整配置指南
使用阿里云 CDN 保护网站真实 IP:完整配置指南 一、宝塔面板准备工作1. 确认网站部署状态2. 宝塔中检查网站配置 二、配置阿里云 CDN1. 添加域名到 CDN2. 配置 DNS 解析3. 配置成功确认 三、宝塔面板安全加固(隐藏 IP 的关键步骤)1. 禁止通过…...
JAVA-StringBuilder使用方法
JAVA-StringBuilder使用方法 常用方法 append(Object obj) 追加内容到末尾 sb.append(" World"); insert(int offset, Object obj) 在指定位置插入内容 sb.insert(5, “Java”); delete(int start, int end) 删除指定范围的字符 sb.delete(0, 5); replace(int start…...
Milvus(9):字符串字段、数字字段
1 字符串字段 在 Milvus 中,VARCHAR 是用于存储字符串数据的数据类型。定义VARCHAR 字段时,有两个参数是必须的: 将datatype 设置为DataType.VARCHAR 。指定max_length ,它定义了VARCHAR 字段可存储的最大字符数。max_length 的有…...
locust压力测试
安装 pip install locust验证是否安装成功 locust -V使用 网上的教程基本上是前几年的,locust已经更新了好几个版本,有点过时了,在此做一个总结 启动 默认是使用浏览器进行设置的 # 使用浏览器 locust -f .\main.py其他参数 Usage: locust […...
Uniapp:showLoading(等待加载)
目录 一、出现场景二、效果展示三、具体使用一、出现场景 在项目的开发中,我们经常会请求后台接口返回数据,但是每一个接口返回数据的时间不一致,有的快,有的慢,这个时候如果不加一个遮罩层,接口返回慢的时候,非常影响用户体验 二、效果展示 三、具体使用 显示加载框…...
线性代数的本质大白话理解
先一句话总结的如下: 线性代数的本质,就是研究“线性变化”——包括空间中点、向量、矩阵之间如何通过线性规则(加法、数乘)变化和联系,并理解这些变化背后的结构。 1. 向量(Vector)——不是数据…...
【Rust通用集合类型】Rust向量Vector、String、HashMap原理解析与应用实战
✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,…...
Kotlin await等待多个异步任务都完成后才进行下一步操作
Kotlin await等待多个异步任务都完成后才进行下一步操作 import kotlinx.coroutines.*fun main() {runBlocking {val tagA "a"val tagB "b"val a async {worker(tagA)}val b async {worker(tagB)}println("${System.currentTimeMillis()} 等待 $t…...
佛山大旺高新区3650 M5 ERP服务器维修案例
1:机器型号:联想system x3650 m5 2:故障问题:纺织公司由于没有专业的it网管,导致服务器各种爆故障灯,本次处理的是用户反馈最近ERP软件使用过程中经常弹出资源不足的报错。 3:于是预约我们工程…...
利用Python生成Xilinx FPGA ROM IP核 .coe初始化文件
以下是一个 Python 脚本,用于生成 Xilinx IP ROM 的.coe 格式初始化文件,假设ROM 深度为 1024,数据位宽为 32bit,使用随机的 32 位无符号数进行初始化: import random# 定义ROM的深度和数据位宽 rom_depth 1024 data…...
配置电子邮件服务
配置电子邮件服务 一.基于Postfix的邮件发送 1. 安装Postfix yum install postfix -y 再下载一个telnet工具 yum -y install telnet 启动Postfix systemctl start postfix systemctl enable postfix 查看系统中端口是否被Postfix使用 netstat -tnlp | gre…...
WGCAT工单系统发现错误 定时处理工单数据任务错误
一直在用WGCAT工单系统,今天在系统日志里,看到了这个错误提示,不知道是什么原因 2025-04-26 07:05:00.000 [taskScheduler-10] INFO com.wgcloud.task.ScheduledTask - 定时处理工单数据任务开始----------2025-04-26 07:05:00 2025-04-26 …...
软件工程(一):黑盒测试与白盒测试
黑盒测试(Black Box Testing) 定义 黑盒测试是指不关心程序内部实现细节,只关注输入和输出的测试方法。把被测软件当作一个“黑盒子”,只依据功能说明书或需求文档来编写测试用例,验证功能是否正确。 特点 不需要了…...
emqx部署
要修改文件-命名空间-节点选择器 #apiVersion: v1 ##kind: ConfigMap ##metadata: ## name: emqx-config ##data: ## emqx.conf: | ## # --- apiVersion: v1 kind: PersistentVolume metadata:name: emqx-pv spec:capacity:storage: 5GivolumeMode: FilesystemaccessMode…...
【KWDB 创作者计划】_KWDB产品技术解读
文章目录 每日一句正能量一、KWDB简介二、官网信息三、技术亮点解读(一)存储引擎(二)查询引擎(三)分布式架构 四、应用场景五、总结 每日一句正能量 你的心为什么这样分散,使得你放慢了脚步。他…...
C++ 表达式求值优先级、结合律与求值顺序(五十九)
1. 运算符优先级与结合律 优先级(Precedence) 决定未加括号时运算符如何“绑”在一起:5 10 * 20 / 2; // 等同于 5 ((10 * 20) / 2)结合律(Associativity) 决定同级运算符的结合方向: 左结合࿰…...
乐理学习笔记(一)---节拍与音符
节拍 衡量音的长度和节奏的基本单位,以强弱关系按照一定的规律循环进行 拍大腿、拍手 类型 上面的这些不同类型节拍的强弱关系中第一个都是强(起确定性作用,而不是音量最大) 强和弱是决定性的区别,每一个强拍是和弦…...
《系统架构 - Java 企业应用架构中的完整层级划分》
文章目录 Java 企业应用架构中的完整层级划分核心层级(基础架构)业务逻辑层接口层基础设施层辅助层级特殊架构层级现代架构扩展层各层调用关系示例分层原则建议 Java 企业应用架构中的完整层级划分 除了常见的 Controller、Service、DAO 等标准层级外&a…...
Adobe Lightroom Classic v14.3.0.8 一款专业的数字摄影后期处理软件
软件介绍 Adobe Lightroom Classic 2025中文激活版(Adobe桌面照片编辑软件)LRC2025(LR2025本地离线版)是一款桌面照片编辑器和相册管理软件的raw格式编辑软件,支持各种RAW图像相机配置,HDR全景照片&#x…...
SQL 易混易错知识点笔记1(drop,role,%,localhost)
DROP 与 DELETE 的区别 DELETE:删除表中的数据行,属于DML操作,可回滚,可带WHERE条件 DELETE FROM table WHERE condition; -- 删除特定行 DELETE FROM table; -- 删除所有行但保留表结构 DROP:删除整个数据库对象(表、…...
C++23 std::bind_back:一种调用包装器 (P2387R3)
文章目录 引言背景知识旧有的绑定工具C20的std::bind_front std::bind_back的定义和功能定义功能 std::bind_back的使用场景简化回调函数部分应用参数 std::bind_back与其他绑定工具的对比与std::bind的对比与std::bind_front的对比 总结 引言 在C的发展历程中,每一…...
使用多线程快速向Excel中快速插入一万条数据案例
当有大量数据需要存入Excel时,使用传统的单线程完成会有以下这些弊端: 导入速度慢:单线程一次只能处理一个任务,在导入大量数据时,需要逐个将数据写入 Excel。这意味着 CPU 在大部分时间里只能处理一个数据块ÿ…...
RestRequest ,newtonsoft解析
var request new RestRequest(Method.GET); IRestResponse response client.Execute(request); Console.WriteLine(response.Content); //保存token Newtonsoft.Json.Linq.JObject obj3 Newtonsoft.Json.Linq.JObject.Pars…...
vs2022解决 此项目需要MFC库。从visual studio安装程序(单个组件选项卡)为正在使用的任何工具和体系结构安装他们问题
使用visual studio 2022创建MFC 单文档的项目,编译器报错: 严重性 代码 说明 项目 文件 行 禁止显示状态 详细信息 错误 MSB8041 此项目需要 MFC 库。从 Visual Studio 安装程序(单个组件选项卡)为正在使用的任何工具集和体系结构安装它们。 osgEarthMFC…...
面试算法高频08-动态规划-03
练习题 题目描述 你是一个专业的小偷,计划偷窃沿街的房屋。每间房内都藏有一定的现金,影响你偷窃的唯一制约因素就是相邻的房屋装有相互连通的防盗系统,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警。 给定一个代表每…...
uniapp做app,使用v-for遍历渲染第二层的时候,打包到手机上渲染不出第二层的数据
1.打包apk要严格注意一点,在data中定义的时候要把第二层定义上, pointspower: [{ jcdbh: 1, cgqbhs:[] }] 不然会出现未定义的情况,直接把二层结构定义上,有利无害 2.渲染…...
Uniapp(vue):生命周期
目录 一、Vue生命周期二、Uniapp中页面的生命周期三、执行顺序比较一、Vue生命周期 setup():是在beforeCreate和created之前运行的,所以可以用setup代替这两个钩子函数。onBeforeMount():已经完成了模板的编译,但是组件还未挂载到DOM上的函数。onMounted():组件挂载到DOM完…...
Git技巧:Git Hook,自动触发,含实战分享
Git技巧:Git Hook,自动触发,含实战分享 最近项目需要1个git合入时触发脚本的功能,使用Git Hook功能实现,总结如下: Git项目在路径:repo\.git\hooks下有很多文件,这些文件就是本地钩…...
DeepSeek创始人梁文峰是个什么样的人?
梁文峰是一位在人工智能领域具有深远影响力的企业家和技术创新者,他的个人经历和成就展现了他作为一位技术天才、创新领袖以及社会责任感强的企业家的多重身份。 从学术背景来看,梁文峰出生于广东湛江吴川,17岁时以高考状元的身份考入浙江大…...