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

如何实现一个分布式锁

如何实现一个分布式锁

本篇内容主要介绍如何使用 Java 语言实现一个注解式的分布式锁,主要是通过注解+AOP 环绕通知来实现。

1. 锁注解

我们首先写一个锁的注解

/*** 分布式锁注解*/
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.METHOD})
@Documented
public @interface RedisLock {long DEFAULT_TIMEOUT_FOR_LOCK = 5L;long DEFAULT_EXPIRE_TIME = 60L;String key() default "your-biz-key";long expiredTime() default DEFAULT_EXPIRE_TIME;long timeoutForLock() default DEFAULT_TIMEOUT_FOR_LOCK;}

expiredTime 是设置锁的过期时间,timeoutForLock 是设置等待锁的超时时间。如果没有等待获得锁的超时时间这个功能,那么其他线程在获取锁失败时只能直接失败,无法进行排队等待。

我们如何使用这个注解呢,很容易,在需要加锁的业务方法上直接用就行.如下,我们有一个库存服务类,它有一个扣减库存方法,该方法将数据库中的一个库存商品的数量减一。在并发场景下,如果我们没有对其进行资源控制,必然会发生库存扣减不一致现象。

public class StockServiceImpl {@RedisLock(key = "stock-lock", expiredTime = 10L, timeoutForLock = 5L)public void deduct(Long stockId) {Stock stock = this.getById(1L);Integer count = stock.getCount();stock.setCount(count - 1);this.updateById(stock);}
}

2. 在 AOP 切面中进行加锁处理

我们需要使用 AOP 来处理什么?自然是处理使用@RedisLock的方法,因此我们写一个切点表达式,它匹配所有标有 @RedisLock 注解的方法。
接着,我们将此切点表达式与 @Around 注解结合使用,以创建环绕通知,在目标方法执行前后执行我们的加锁解锁逻辑。

因此,基本的逻辑我们就理清了,代码大致长下面这个样子:

public class RedisLockAspect {private final RedisTemplate<String, Object> redisTemplate;// 锁的redis key前缀private static final String DEFAULT_KEY_PREFIX = "lock:";// 匹配所有标有 @RedisLock 注解的方法@Pointcut("@annotation(com.kelton.lock.annotation.RedisLock)")public void lockAnno() {}@Around("lockAnno()")public void invoke(ProceedingJoinPoint joinPoint) throws Exception {// 获取拦截方法上的RedisLock注解RedisLock annotation = getLockAnnotationOnMethod(joinPoint);// 获取锁keyString key = getKey(annotation);// 锁过期时间long expireTime = annotation.expiredTime();// 获取锁的等待时间long timeoutForLock = annotation.timeoutForLock();// 在这里加锁someCodeForLock...// 执行业务joinPoint.proceed();// 在这里解锁someCodeForUnLock...}

我们在加锁的时候,需要用上 timeoutForLock 这个属性,我们通过自旋加线程休眠的方式,来达到在一段时间内等待获取锁的目的。如果自旋时间结束后,还没获取锁,则抛出异常,这里可以根据自己情况而定。自旋加锁代码如下:

    // 自旋获取锁long endTime = System.currentTimeMillis() + timeoutForLock * 1000;boolean acquired = false;String uuid = UUID.randomUUID().toString();while(System.currentTimeMillis() < endTime) {Boolean absent = redisTemplate.opsForValue().setIfAbsent(key, uuid, expireTime, TimeUnit.SECONDS);if (Boolean.TRUE.equals(absent)) {acquired = true;break;} else {// 获取不到锁,尝试休眠100毫秒后重试Thread.sleep(100);}}// 超时未获取到锁, 抛出异常,可根据自己业务而定if (!acquired) {throw new RuntimeException("获取锁异常");}

我们发现上面加锁的时候设置了一个 uuid 作为 value 值,这是为了在锁释放的时候,不误删其他线程上的锁,随后,我们就可以执行被 AOP 切中的方法,执行结束释放锁。代码如下:

    try {// 执行业务joinPoint.proceed();} catch (Throwable e) {log.error("业务执行出错!");} finally {// 解锁时进行校验,只删除自己线程加的锁String value = (String) redisTemplate.opsForValue().get(key);if (uuid.equals(value)) {redisTemplate.delete(key);} else {log.warn("锁已过期!");}}

到这里,我们就以注解+AOP 的方式实现了分布式锁的功能。当然,以上只实现了分布式锁的简单功能,还缺少了分布式锁的 key 自动续约防止锁过期功能,以及锁重入功能。

目前,RedisLockAspect的完整代码如下:

@Component
@Aspect
@Slf4j
@AllArgsConstructor
public class RedisLockAspect {// 匹配所有标有 @RedisLock 注解的方法@Pointcut("@annotation(com.kelton.lock.annotation.RedisLock)")public void lockAnno() {}@Around("lockAnno()")public void invoke(ProceedingJoinPoint joinPoint) throws Exception {// 获取拦截方法上的RedisLock注解RedisLock annotation = getLockAnnotationOnMethod(joinPoint);String key = getKey(annotation);// 锁过期时间long expireTime = annotation.expiredTime();// 获取锁的等待时间long timeoutForLock = annotation.timeoutForLock();// 自旋获取锁long endTime = System.currentTimeMillis() + timeoutForLock * 1000;boolean acquired = false;String uuid = UUID.randomUUID().toString();while(System.currentTimeMillis() < endTime) {Boolean absent = redisTemplate.opsForValue().setIfAbsent(key, uuid, expireTime, TimeUnit.SECONDS);if (Boolean.TRUE.equals(absent)) {acquired = true;break;} else {// 获取不到锁,尝试休眠100毫秒后重试Thread.sleep(100);}}// 超时未获取到锁, 抛出异常,可根据自己业务而定if (!acquired) {throw new RuntimeException("获取锁异常");}try {// 执行业务joinPoint.proceed();} catch (Throwable e) {log.error("业务执行出错!");} finally {// 解锁时进行校验,只删除自己线程加的锁String value = (String) redisTemplate.opsForValue().get(key);if (uuid.equals(value)) {redisTemplate.delete(key);} else {log.warn("锁已过期!");}}}private String getKey(RedisLock redisLock) {if (Objects.isNull(redisLock)) {return DEFAULT_KEY_PREFIX + "default";}return DEFAULT_KEY_PREFIX + redisLock.key();}private RedisLock getLockAnnotationOnMethod(ProceedingJoinPoint joinPoint) {MethodSignature signature = (MethodSignature) joinPoint.getSignature();Method method = signature.getMethod();return method.getAnnotation(RedisLock.class);}}

3. key 自动续约防止锁过期

我们接着完善该分布式锁,为其添加 key 自动续约防止锁过期的功能。我们的思路与Redission的watch dog类似,开启一个后台线程,来定时检查需要续约的锁。我们如何判断一个锁是否需要续约呢,我们可以简单定义一个续约分界线,比如在锁过期时间的三分之二的时间点及之后,对锁进行续约。

3.1 定义一个续约任务4

我们来定义一个锁续约任务,那我们需要什么信息呢?
我们至少需要锁的 key,锁要设置的过期时间。这是两个最基本的信息。
要判断在锁过期时间的三分之二的时间点及之后进行续约,那么我们还需要记录锁上次续约的时间点。
此外,我们还可以为锁续约任务添加最大续约次数限制,这可以避免某些执行时间特别久的任务不断占用锁。所以我们还需要记录当前锁续约次数和最大续约次数。
对超过最大续约次数的锁的线程,我们直接将其停止,因此我们也记录一下该锁的线程。
结合上面的分析,我们定义的锁续约任务类如下:

public class LockRenewTask {/*** key*/private final String key;/*** 过期时间。单位:秒*/private final long expiredTime;/*** 锁的最大续约次数*/private final int maxRenewCount;/*** 锁的当前续约次数*/private int currentRenewCount;/*** 最新更新时间*/private LocalDateTime latestRenewTime;/*** 业务线程*/private final Thread thread;public LockRenewTask(String key, long expiredTime, int maxRenewCount, Thread thread) {this.key = key;this.expiredTime = expiredTime;this.maxRenewCount = maxRenewCount;this.thread = thread;this.latestRenewTime = LocalDateTime.now();}/*** 是否到达续约时间* @return*/public boolean isTimeToRenew() {LocalDateTime now = LocalDateTime.now();Duration duration = Duration.between(latestRenewTime, now);return duration.toSeconds() >= ((double)(this.expiredTime / 3) * 2);}/*** 是否达到最大续约次数* @return*/public boolean exceedMaxRenewCount() {return this.currentRenewCount >= this.maxRenewCount;}public synchronized void renew() {this.currentRenewCount++;this.latestRenewTime = LocalDateTime.now();}// 取消业务方法public void cancel() {thread.interrupt();}public String getKey() {return key;}public long getExpiredTime() {return expiredTime;}
}

我们添��了一些关于锁续约的方法:

  • isTimeToRenew(): 判断是否可以对锁进行续约
  • exceedMaxRenewCount(): 判断是否达到最大续约次数
  • renew(): 来标记一次续约操作
  • cancel(): 取消业务方法
3.2 定义一个锁续约任务处理器

接着,我们定义一个定时执行该续约任务的 handler。该 handler 也比较简答,核心逻辑是持有一个类型为 List<LockRenewTask>taskList 来添加续约任务,且使用一个 ScheduledExecutorService 来定时遍历该 taskList 来执行续约任务。该 handler 再对外暴露一个 addRenewTask 方法,方便外部调用来添加续约任务到 taskList 中。

@Slf4j
@Component
public class LockRenewHandler {@Autowiredprivate RedisTemplate<String, Object> redisTemplate;/*** 保障对 taskList的添加删除操作是线程安全的*/private final ReentrantLock taskListLock = new ReentrantLock();private final List<LockRenewTask> taskList = new ArrayList<>();private final ScheduledExecutorService taskExecutorService;{taskExecutorService = Executors.newScheduledThreadPool(Runtime.getRuntime().availableProcessors());taskExecutorService.scheduleAtFixedRate(() -> {try {executeRenewTask();} catch (Exception e) {//错误处理}}, 1, 2, TimeUnit.SECONDS);}/*** 添加续约任务*/public void addRenewTask(LockRenewTask task) {taskListLock.lock();try {taskList.add(task);} finally {taskListLock.unlock();}}/*** 执行续约任务*/private void executeRenewTask() {log.info("开始执行续约任务");if (CollectionUtils.isEmpty(taskList)) {return;}// 需要删除的任务,暂存这个集合中  取消List<LockRenewTask> cancelTask = new ArrayList<>();// 获取任务副本List<LockRenewTask> copyTaskList = new ArrayList<>(taskList);for (LockRenewTask task : copyTaskList) {try {// 判断 Redis 中是否存在 keyif (!redisTemplate.hasKey(task.getKey())) {cancelTask.add(task);continue;}// 大于等于最大续约次数if (task.exceedMaxRenewCount()) {// 停止续约任务task.cancel();cancelTask.add(task);continue;}// 到达续约时间if (task.isTimeToRenew()) {log.info("续约任务:{}", task.getKey());redisTemplate.expire(task.getKey(), task.getExpiredTime(), TimeUnit.SECONDS);task.renew();}} catch (Exception e) {//错误处理log.error("处理任务出错:{}", task);}}// 加锁,删除 taskList 中需要移除的任务taskListLock.lock();try {taskList.removeAll(cancelTask);// 清理cancelTask,避免堆积,产生内存泄露cancelTask.clear();} finally {taskListLock.unlock();}}
}

总结一下 LockRenewHandler的主要作用:它负责管理和执行续约任务,以延长 Redis 中键的过期时间。

  • 添加续约任务:addRenewTask() 方法允许添加新的续约任务到内部列表 taskList 中。
  • 执行续约任务:executeRenewTask() 方法定期执行续约任务。它检查每个任务的状态,并根据需要续约 Redis 中的键。
  • 移除完成的任务:维护一个 cancelTask 列表,用于存储需要从 taskList 中移除的任务。在 executeRenewTask() 方法中,它会将完成的任务添加到 cancelTask 列表中,并在之后将其从 taskList 中移除。

大概的工作流程如下:

  • 续约任务被添加到 taskList 中。

  • executeRenewTask() 方法定期执行,它检查每个任务的状态:

    • 如果 Redis 中不再存在该键,则取消任务。
    • 如果任务的续约次数达到上限,则取消任务。
    • 如果是时候续约了,则续约 Redis 中的键并更新任务的续约次数,记录续约时间点。
  • 完成的任务被添加到 cancelTask 列表中。

  • executeRenewTask() 方法获取 taskList 的副本,并从副本中移除 cancelTask 中的任务,并且在完成移除任务操作后清空cancelTask

  • 更新后的 taskList 被保存回类中。

两个需要注意的点

  • 我们遍历taskList时拷贝了一份副本进行遍历,因为taskList是可变的,这样可以避免在遍历的时候产生并发修改问题。
  • cancelTask需要清理,避免产生内存泄漏。

通过这种方式,LockRenewHandler 可以确保 Redis 中的键在需要时得到续约,并自动移除完成或失败的任务。

3.3 添加锁续约任务

在上面 3.1 节和 3.2 节我们定义好了锁续约任务和处理锁续约任务的核心代码,接下来我们需要在第 2 节加锁解锁的 AOP 处理逻辑上进行一点小小的修改,主要就是在执行加锁之后,执行业务代码之前,添加上锁续约任务。修改位置如下:

public void invoke(ProceedingJoinPoint joinPoint) throws Exception {... // 省略代码try {// 添加锁续约任务LockRenewTask task = new LockRenewTask(key, annotation.expiredTime(), annotation.maxRenew(), Thread.currentThread());lockRenewHandler.addRenewTask(task);log.info("添加续约任务, key:{}", key);// 执行业务joinPoint.proceed();} catch (Throwable e) {log.error("业务执行出错!");} finally {// 解锁时进行校验,只删除自己线程加的锁String value = (String) redisTemplate.opsForValue().get(key);if (uuid.equals(value)) {redisTemplate.delete(key);} else {log.warn("锁已过期!");}}... // 省略代码
}

到这里,我们的分布式锁已经相当完善了,把锁自动续约的功能也加上了。当然,还没有实现锁的可重入性。

相关文章:

如何实现一个分布式锁

如何实现一个分布式锁 本篇内容主要介绍如何使用 Java 语言实现一个注解式的分布式锁&#xff0c;主要是通过注解AOP 环绕通知来实现。 1. 锁注解 我们首先写一个锁的注解 /*** 分布式锁注解*/ Retention(RetentionPolicy.RUNTIME) Target({ElementType.METHOD}) Documente…...

Ajax从零到实战

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 非常期待和您一起在这个小…...

编程参考 - 在C++移动构造函数声明中使用noexcept

在 C 中&#xff0c;noexcept 是用于表示函数不抛出异常的指定符。它既可用于常规函数&#xff0c;也可用于特殊成员函数&#xff0c;包括构造函数和析构函数。使用 noexcept 可以帮助编译器进行优化&#xff0c;提高代码的安全性和正确性。 In C, noexcept is a specifier use…...

Vue2/Vue3实现全局/局部添加防篡改水印的效果。删除元素无效!更改元素属性无效!支持图片、元素、视频等等。

水印目的 版权保护:水印可以在图片、文档或视频中嵌入作者、品牌或版权所有者的信息,以防止未经授权的复制、传播或使用。当其他人使用带有水印的内容时,可以追溯到原始作者或版权所有者,从而加强版权保护。 身份识别:水印可以用作作者或品牌的标识符,使观众能够轻松识…...

GuLi商城-商品服务-API-属性分组-获取分类属性分组

获取分类属性分组接口开发 操作的是这张表 造数据: 后台代码: @Override public PageUtils queryPage(Map<String, Object> params, Long catelogId) {//select * from pms_attr_group where catelog_id=? and (attr_group_id=key or attr_group_name like %key%)Stri…...

安全测试理论

安全测试理论 什么是安全测试&#xff1f; 安全测试&#xff1a;发现系统安全隐患的过程安全测试与传统测试区别 传统测试&#xff1a;发现bug为目的 安全测试&#xff1a;发现系统安全隐患什么是渗透测试 渗透测试&#xff1a;已成功入侵系统为目标的的攻击过程渗透测试与安全…...

序列化和反序列化

面试题&#xff1a;对序列化和反序列化的理解&#xff1f; 我们之所以需要序列化&#xff0c;它核心的目的是为了解决网络通信之间的对象传输的问题&#xff0c;也就是说&#xff0c;如何把当前JVM进程的一个对象&#xff0c;通过跨网络传输到另一个JVM进程里面&#xff0c;而序…...

OpenCV中使用Canny算法在图像中查找边缘

操作系统&#xff1a;ubuntu22.04OpenCV版本&#xff1a;OpenCV4.9IDE:Visual Studio Code编程语言&#xff1a;C11 算法描述 Canny算法是一种广泛应用于计算机视觉和图像处理领域中的边缘检测算法。它由John F. Canny在1986年提出&#xff0c;旨在寻找给定噪声条件下的最佳边…...

基于springboot+vue+uniapp的机电公司管理信息系统

开发语言&#xff1a;Java框架&#xff1a;springbootuniappJDK版本&#xff1a;JDK1.8服务器&#xff1a;tomcat7数据库&#xff1a;mysql 5.7&#xff08;一定要5.7版本&#xff09;数据库工具&#xff1a;Navicat11开发软件&#xff1a;eclipse/myeclipse/ideaMaven包&#…...

电子期刊制作实战教程:从零开始制作

​随着互联网的普及&#xff0c;电子期刊已经成为了信息传递的重要载体。它以便捷、环保、互动性强等特点受到了越来越多人的青睐。那么&#xff0c;如何从零开始制作一份吸引人的电子期刊呢&#xff1f; 1.要制作电子杂志,首先需要选择一款适合自己的软件。比如FLBOOK在线制作…...

11.FreeRTOS_事件组

事件组概述 事件组的作用&#xff1a; 可以等待某一个事件发生可以等待若干个事件发生可以等待若干个事件中的某一个事件发生 同步点是事件组的另一个使用方式&#xff0c;它可以让多个任务进行阻塞等待&#xff0c;当全部事件完成后&#xff0c;再一起解除任务的阻塞。常常…...

Python爬虫-爬取三国演义文本数据-bs4

bs4进行数据解析 -数据解析的原理: - 1.标签定位 -2.提取标签、标签属性中存储的数据值 - bs4数据解析的原理: - 1.实例化一个BeautifulSoup对象,并且将页面源码数据加载到该对象中 -2.通过调用BeautifulSoup对象中相关的属性或者方法进行标签定位和数据提取 - 环境安装: - pi…...

html5——列表、表格

目录 列表 无序列表 有序列表 自定义列表 表格 基本结构 示例 表格的跨列 表格的跨行 列表 无序列表 <ul>【声明无序列表】 <li>河间驴肉火烧</li>【声明列表项】 <li>唐山棋子烧饼</li> <li>邯郸豆沫</li> <l…...

【Python字符串攻略】:玩转文字,编织程序的叙事艺术

文章目录 &#x1f680;一.字符串基础&#x1f308;二.查看数据类型⭐三.转化❤️四.字符串索引&#x1f6b2;五.字符串切片&#x1f3ac;六.字符串切片-步长☔七.反向切片注意事项&#x1f6b2;八.字符串&#x1f4a5;查&#x1f4a5;改&#x1f4a5;删 ❤️九.字符串拼接&…...

element form表单中密码框被自动赋值,并默认背景色为白色,手动输值后背景色才是自己配置的背景色,与表单的自动填充有关

事件背景&#xff1a; 一个表单&#xff0c;有两组需要输入密码的地方&#xff0c;两组都被填充用户名密码&#xff0c;其中一组是其他信息&#xff0c;不是用户名密码&#xff0c;也被填充了&#xff0c;且input背景色是白色&#xff0c;表单中的input已经手动配置为无背景色&…...

【UE5.1 角色练习】15-枪械射击——子弹发射物

目录 效果 步骤 一、创建并发射子弹 二、优化子弹 效果 步骤 一、创建并发射子弹 1. 在前面的文章中&#xff08;【UE5.1 角色练习】06-角色发射火球-part1&#xff09;我们创建了蓝图“BP_Skill_FireBall” 这里我们复制一份命名为“BP_Ammo_5mm”&#xff0c;用于表示…...

Zynq7000系列FPGA中的DMA控制器的编程限制

有关DMAC编程时适用的限制信息&#xff0c;有四个考虑因素&#xff1a; 固定非对齐突发Endian swap size restrictions&#xff1a;在数据传输或处理过程中&#xff0c;不同字节序&#xff08;Endian&#xff09;之间的转换和对应的限制在DMA周期内更新通道控制寄存器当MFIFO满…...

超简易高效的 AI绘图工具—与sd-webui一致界面,6G显存最高提升75%出图速率!(附安装包)

大家好&#xff0c;我是灵魂画师向阳 今天给大家分享一个基于Stable Diffusion WebUI 构建的AI绘图工具—sd-webui-forge&#xff0c;该工具的目标在于简化插件开发&#xff0c;优化资源管理&#xff0c;加速推理。 Forge承诺永远不会对Stable Diffusion WebUI用户界面添加不…...

ArduPilot开源代码之OpticalFlow_backend

ArduPilot开源代码之OpticalFlow_backend 1. 源由2. Library设计3. 重要例程3.1 OpticalFlow_backend::_update_frontend3.2 OpticalFlow_backend::_applyYaw 4. 总结5. 参考资料 1. 源由 光流计是一种低成本定位传感器&#xff0c;所有的光流计设备传感驱动代码抽象公共部分统…...

设计模式探索:适配器模式

1. 适配器模式介绍 1.1 适配器模式介绍 适配器模式&#xff08;adapter pattern&#xff09;的原始定义是&#xff1a;将一个类的接口转换为客户期望的另一个接口&#xff0c;适配器可以让不兼容的两个类一起协同工作。 适配器模式的主要作用是把原本不兼容的接口&#xff0c…...

在鸿蒙HarmonyOS 5中实现抖音风格的点赞功能

下面我将详细介绍如何使用HarmonyOS SDK在HarmonyOS 5中实现类似抖音的点赞功能&#xff0c;包括动画效果、数据同步和交互优化。 1. 基础点赞功能实现 1.1 创建数据模型 // VideoModel.ets export class VideoModel {id: string "";title: string ""…...

【Java学习笔记】Arrays类

Arrays 类 1. 导入包&#xff1a;import java.util.Arrays 2. 常用方法一览表 方法描述Arrays.toString()返回数组的字符串形式Arrays.sort()排序&#xff08;自然排序和定制排序&#xff09;Arrays.binarySearch()通过二分搜索法进行查找&#xff08;前提&#xff1a;数组是…...

《从零掌握MIPI CSI-2: 协议精解与FPGA摄像头开发实战》-- CSI-2 协议详细解析 (一)

CSI-2 协议详细解析 (一&#xff09; 1. CSI-2层定义&#xff08;CSI-2 Layer Definitions&#xff09; 分层结构 &#xff1a;CSI-2协议分为6层&#xff1a; 物理层&#xff08;PHY Layer&#xff09; &#xff1a; 定义电气特性、时钟机制和传输介质&#xff08;导线&#…...

【机器视觉】单目测距——运动结构恢复

ps&#xff1a;图是随便找的&#xff0c;为了凑个封面 前言 在前面对光流法进行进一步改进&#xff0c;希望将2D光流推广至3D场景流时&#xff0c;发现2D转3D过程中存在尺度歧义问题&#xff0c;需要补全摄像头拍摄图像中缺失的深度信息&#xff0c;否则解空间不收敛&#xf…...

ArcGIS Pro制作水平横向图例+多级标注

今天介绍下载ArcGIS Pro中如何设置水平横向图例。 之前我们介绍了ArcGIS的横向图例制作&#xff1a;ArcGIS横向、多列图例、顺序重排、符号居中、批量更改图例符号等等&#xff08;ArcGIS出图图例8大技巧&#xff09;&#xff0c;那这次我们看看ArcGIS Pro如何更加快捷的操作。…...

智能分布式爬虫的数据处理流水线优化:基于深度强化学习的数据质量控制

在数字化浪潮席卷全球的今天&#xff0c;数据已成为企业和研究机构的核心资产。智能分布式爬虫作为高效的数据采集工具&#xff0c;在大规模数据获取中发挥着关键作用。然而&#xff0c;传统的数据处理流水线在面对复杂多变的网络环境和海量异构数据时&#xff0c;常出现数据质…...

【C++特殊工具与技术】优化内存分配(一):C++中的内存分配

目录 一、C 内存的基本概念​ 1.1 内存的物理与逻辑结构​ 1.2 C 程序的内存区域划分​ 二、栈内存分配​ 2.1 栈内存的特点​ 2.2 栈内存分配示例​ 三、堆内存分配​ 3.1 new和delete操作符​ 4.2 内存泄漏与悬空指针问题​ 4.3 new和delete的重载​ 四、智能指针…...

【Android】Android 开发 ADB 常用指令

查看当前连接的设备 adb devices 连接设备 adb connect 设备IP 断开已连接的设备 adb disconnect 设备IP 安装应用 adb install 安装包的路径 卸载应用 adb uninstall 应用包名 查看已安装的应用包名 adb shell pm list packages 查看已安装的第三方应用包名 adb shell pm list…...

日常一水C

多态 言简意赅&#xff1a;就是一个对象面对同一事件时做出的不同反应 而之前的继承中说过&#xff0c;当子类和父类的函数名相同时&#xff0c;会隐藏父类的同名函数转而调用子类的同名函数&#xff0c;如果要调用父类的同名函数&#xff0c;那么就需要对父类进行引用&#…...

API网关Kong的鉴权与限流:高并发场景下的核心实践

&#x1f525;「炎码工坊」技术弹药已装填&#xff01; 点击关注 → 解锁工业级干货【工具实测|项目避坑|源码燃烧指南】 引言 在微服务架构中&#xff0c;API网关承担着流量调度、安全防护和协议转换的核心职责。作为云原生时代的代表性网关&#xff0c;Kong凭借其插件化架构…...