简洁易懂:源码+实战讲解Redisson并发锁及看门狗自动续期
1 缘起
有一次同事问Redisson存储的键是否为hash?
我当时,没有看Redisson的相关源码,只知道应用,
所以没有办法回答,于是开始看看Redisson实现的源码,
顺便写了一个单机Redisson测试,
发现Redisson有看门狗功能,但是,不是所有的方法都会触发,
只有lock()方法才会触发看门狗,其他的方法不会触发看门狗功能,
如lock(long, TimeUnit),tryLock(),
本文从结果触发,先理论分析,然后,代码实践验证。
先从源码讲解Redisson相关的知识,包括如何新建锁、释放锁、看门狗功能,
然后代码实战,验证这些功能,
感兴趣的同学可以手动调试。
帮助读者轻松应对知识交流与考核。
版本信息:
SpringBoot:2.1.7
Redisson:3.7.5
2 Redisson属性源码分析与验证
2.1 新建锁
位置:org.redisson.RedissonLock#tryLockInnerAsync
新建锁和可重入源码如下图所示,由源码可知,
Redisson使用Lua脚本新建锁。
- 为什么?
保证操作原子性。
因为新建数据和添加过期时间是两步操作,不能保证原子性,
因此使用Lua实现,保证新建锁时是原子操作。
Redisson实现的锁是可重入的。同一个线程获取锁,无需重新排队申请,只需增加锁的持有次数。
- 怎么做?
判断锁是的属性是否存在。
通过hexists判断锁的属性是否存在(UUID:threadId),如果存在,值+1,同时,更新锁过期时间,
使用Lua脚本,保证原子性。
Lua脚本中的参数KEYS[1],ARGV[1], ARGV[2],
- 分别对应哪些参数呢?
KEYS[1]:Collections.singletonList(getName()),即锁的键名;
ARGV[1]:internalLockLeaseTime,即锁过期时间;
ARGV[2]:getLockName(threadId),即锁(Hash)属性的键,UUDI:threadId。
其中,getLockName源码如下图所示,由图可知,UUID拼接threadId作为锁属性的键。
顺便补充一下,Hash结构,键名,即获取该Hash的键,属性键名,即Hash内部数据的键。
下面看一下新建锁的调试过程,如下图所示,
(单步调试,如果操作不及时或者有其他时延,会导致Redis数据过期,因此,选择单步调试)
由图可知,新建的锁,键名称为redisLockKey,
默认锁过期时间为30秒,UUID为edb0f8cb-ec11-4b29-a107-12be152c4a5e,
threadId为71,锁属性的键为:edb0f8cb-ec11-4b29-a107-12be152c4a5e:71
,属性键的值为1。
新建锁之后,会在Redis生成对应的数据,
如下图所示,由图可知,Redisson新建的锁键为redisLockKey,
锁属性名称为:edb0f8cb-ec11-4b29-a107-12be152c4a5e:71
,属性值为1。
过期时间TTL已经开始倒计时了,实时的值为17秒。
2.2 获取锁
Redisson获取锁有两种方式:lock获取和tryLock获取。
lock()获取锁,如果得不到,会一直等,同时,lock()会开启看门狗功能,在看门狗巡查期间,锁不会过期;
lock(long, java.util.concurrent.TimeUnit)获取锁,会一直等,当锁过期后,即可自动获取;
tryLock()获取锁,得不到会一直等,当锁过期后,可自动获取到;
tryLock(long, long, java.util.concurrent.TimeUnit)获取锁,有两个时间参数,可以同时指定等待锁的最大时间,以及锁过期时间,如果等待最大时间大于锁过期时间,则锁被提前释放,重新生成锁;
2.2.1 lock
位置:java.util.concurrent.locks.Lock#lock
无参的lock方法来自JUC接口,
Redisson自身实现lock,并加入看门狗功能。
无返回值。
Redisson的实现源码如下图所示,这里给出路径,感兴趣可自行查看。
位置:org.redisson.RedissonLock#lock()
有参的lock源码如下图所示,
这个lock来自Redisson的RLock,添加了锁过期时间,
可以自定义锁的过期时间,这个获取锁的方法不会触发看门狗。
无返回值。
位置:org.redisson.api.RLock#lock
Redisson实现源码如下,这里给出路径,感兴趣可自行查看。
位置:org.redisson.RedissonLock#lock(long, java.util.concurrent.TimeUnit)
2.2.2 tryLock
位置:java.util.concurrent.locks.Lock#tryLock()
获取锁,无法获取时等待,当锁自动过期后,可自动获取。
有返回值,可通过返回值判断是否获得锁。
true:成功获取锁;
false:未获取锁。
Redisson实现tryLock源码如下图所示。
位置:org.redisson.RedissonLock#tryLock()
有参数的tryLock源码下图所示,有源码可知,
可指定最大等待锁时间、锁过期时间,
如果等待锁的最大时间小于锁过期时间,返回false,标识未得到锁。
如果等待锁的最大时间大于锁过期时间,等锁释放后,可自动获取锁,返回true,标识得到锁。
有返回值,可通过返回值判断是否获得锁。
true:成功获取锁;
false:未获取锁。
位置:org.redisson.api.RLock#tryLock
获取锁的实现源码如下图所示,这里给出路径,感兴趣可自行查看。
位置:org.redisson.RedissonLock#tryLock(long, long, java.util.concurrent.TimeUnit)
2.3 释放锁
位置:org.redisson.RedissonLock#unlockInnerAsync
释放锁源码如下图所示,由图可知,释放锁时会将锁属性的值-1,(加-1),
当释放的锁存在时,会将锁属性的值减1,减后的值大于零,重置锁的过期时间,保证锁的过期时间周期,
如果减1之后,锁属性值为0,即没有线程再持有锁,则删除该锁(del)。
2.4 锁续期
看门狗
位置:org.redisson.config.Config#lockWatchdogTimeout
看门狗的默认超时时间为30秒,即锁过期时间的续期为30秒,
看门狗的巡查周期为10秒,即每10秒更新一次锁的过期时间,设置为30秒。
两个是独立的时间系统。
30秒:锁的过期时间;
10秒:看门狗巡查周期。
这个10秒是如何得到的:30/3=10。
位置:org.redisson.RedissonLock#scheduleExpirationRenewal
看门狗进行锁续期的源码如下图所示,由源码可知,
当锁存在时,会延迟10秒执行更新锁过期时间,过期时间为30秒,
看门狗的巡查周期就是这个延迟时间:10秒。
当然,好奇的同学会问,锁的默认过期时间怎么就是30秒,
这个30秒是从哪里来的呢?
过期时间(internalLockLeaseTime)默认值取自看门狗的超时时间:30秒,上文有看门狗的默认值源码。
3 实战
版本信息:
SpringBoot:2.1.7
Redisson:3.7.5
3.1 依赖
<!--Redisson-->
<dependency><groupId>org.redisson</groupId><artifactId>redisson</artifactId><version>3.7.5</version>
</dependency>
3.2 配置
3.2.1 Redis配置
application-dev.yml
spring:profiles: devredis:host: localhostport: 6379# 连接超时时间:毫秒timeout: 60000database: 0lettuce:pool:# 连接池最大连接数量max-active: 8# 连接池最大阻塞等待时间:-1无限制max-wait: 60000# 连接池最大空闲连接数量max-idle: 8# 连接池最小空闲连接数量min-idle: 0
3.2.2 绑定Redis配置
package com.hardsoft.monkeyrun.common.lock;import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Configuration;/*** Redis连接参数.** @author xindaqi* @since 2021/4/2 11:41*/
@Configuration
@ConfigurationProperties(prefix="spring.redis")
public class RedisConnectionParams {/*** 主机名*/private String host;/*** 主机端口*/private String port;/*** Redis连接超时时间:毫秒*/private int timeout;/*** Redis数据库编号*/private int database;public void setHost(String host) {this.host = host;}public String getHost() {return host;}public void setPort(String port) {this.port = port;}public String getPort() {return port;}public void setTimeout(int timeout) {this.timeout = timeout;}public int getTimeout() {return timeout;}public void setDatabase(int database) {this.database = database;}public int getDatabase() {return database;}@Overridepublic String toString() {return "RedisConnectionParams{" +"host='" + host + '\'' +", port='" + port + '\'' +", timeout=" + timeout +", database=" + database +'}';}
}
3.2.3 Redisson配置
package com.hardsoft.monkeyrun.common.config;import com.hardsoft.monkeyrun.common.lock.RedisConnectionParams;
import com.hardsoft.monkeyrun.common.lock.RedissonLocker;
import org.redisson.Redisson;
import org.redisson.api.RedissonClient;
import org.redisson.config.Config;
import org.redisson.config.SingleServerConfig;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;import javax.annotation.Resource;import static com.hardsoft.monkeyrun.common.constrant.StringConstant.COLON;
import static com.hardsoft.monkeyrun.common.constrant.StringConstant.REDISSON_SCHEMA;/*** Redisson配置.** @author xindaqi* @since 2021/4/2 11:34*/
@Configuration
public class RedissonConfig {@ResourceRedisConnectionParams redisConnectionParams;/*** 单机模式RedissonClient** @return*/@Bean(destroyMethod = "shutdown")public RedissonClient redissonClient() {Config config = new Config();SingleServerConfig singleServerConfig = config.useSingleServer();StringBuilder redisAddress = new StringBuilder();redisAddress.append(REDISSON_SCHEMA).append(redisConnectionParams.getHost()).append(COLON).append(redisConnectionParams.getPort());singleServerConfig.setAddress(redisAddress.toString());return Redisson.create(config);}@Beanpublic RedissonLocker redissonLocker(RedissonClient redissonClient) {return new RedissonLocker(redissonClient);}}
3.2.4 Redisson获取锁封装
package com.hardsoft.monkeyrun.common.lock;import com.hardsoft.monkeyrun.common.constrant.BooleanConstant;
import org.redisson.api.RLock;
import org.redisson.api.RedissonClient;
import org.springframework.stereotype.Component;import javax.annotation.Resource;
import java.util.concurrent.TimeUnit;/*** Redisson锁** @author xindaqi* @since 2021/3/30 19:01*/
@Component
public class RedissonLocker {RedissonClient redissonClient;public RedissonLocker(RedissonClient redissonClient) {this.redissonClient = redissonClient;}/*** 获取锁* 如果锁不可用,则当前线程处于休眠状态,直到获得锁为止** @param lockKey 锁键*/public void lock(String lockKey) {RLock lock = redissonClient.getLock(lockKey);lock.lock();}/*** 获取锁* 如果锁不可用,则当前线程处于休眠状态,直到获得锁为止,* 如果获取到锁后,执行结束后解锁或达到超时时间后自动释放锁** @param lockKey 锁键* @param timeout 超时时间*/public void lock(String lockKey, int timeout) {RLock lock = redissonClient.getLock(lockKey);lock.lock(timeout, TimeUnit.SECONDS);}/*** 获得锁* 如果锁不可用,则当前线程处于休眠状态,直到获得锁后,* 执行结束后解锁或达到超时时间后会自动释放锁** @param lockKey 锁键* @param unit 时间单位* @param timeout 超时时间*/public void lock(String lockKey, TimeUnit unit, int timeout) {RLock lock = redissonClient.getLock(lockKey);lock.lock(timeout, unit);}/*** 释放锁** @param lockKey 锁键*/public void unlock(String lockKey) {RLock lock = redissonClient.getLock(lockKey);lock.unlock();}/*** 尝试获取锁* 获取到立即返回True* 未获取到返回False** @param lockKey 锁键* @return 获取锁标志位,True:成功获取锁,False:未获取锁*/public boolean tryLock(String lockKey) {RLock lock = redissonClient.getLock(lockKey);return lock.tryLock();}/*** 尝试获取锁* 在等待时间内获取到锁则返回True,否则返回False* 获取到锁,两种执行逻辑:* 1.执行之后,释放锁* 2.达到释放时间leaseTime后,释放锁** @param lockKey 锁键* @param waitTime 等待时间* @param leaseTime 释放时间* @param unit 时间单位* @return 获取锁标志位,True:成功获取锁,False:未获取锁* @throws InterruptedException*/public boolean tryLock(String lockKey, long waitTime, long leaseTime, TimeUnit unit) throws InterruptedException {RLock lock = redissonClient.getLock(lockKey);return lock.tryLock(waitTime, leaseTime, unit);}/*** 判断锁是否被任意一个线程持有** @param lockKey 锁键* @return 锁持有标志位,True:锁被线程持有,False:锁没有被线程持有*/public boolean isLocked(String lockKey) {RLock lock = redissonClient.getLock(lockKey);return lock.isLocked();}public void setRedissonClient(RedissonClient redissonClient) {this.redissonClient = redissonClient;}
}
3.3 接口测试
package com.hardsoft.monkeyrun.api.facade.lock;import com.hardsoft.monkeyrun.common.constrant.BooleanConstant;
import com.hardsoft.monkeyrun.common.constrant.DigitalConstant;
import com.hardsoft.monkeyrun.common.enums.BizExceptionCodeEnums;
import com.hardsoft.monkeyrun.common.exception.BizException;
import com.hardsoft.monkeyrun.common.lock.RedissonLocker;
import com.hardsoft.monkeyrun.common.response.Response;import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;import javax.annotation.Resource;import org.slf4j.Logger;
import org.slf4j.LoggerFactory;import java.util.concurrent.TimeUnit;import static com.hardsoft.monkeyrun.common.constrant.StringConstant.REDIS_LOCK_KEY;/*** 锁测试** @author xindaqi* @since 2021-04-03 11:28:18*/
@RestController
@RequestMapping("/v1/lock")
public class LockTest {private static final Logger logger = LoggerFactory.getLogger(LockTest.class);private final String STOCK_WITH_REDISSON = "stock_with_redisson";private final String STOCK_WITHOUT_REDISSON = "stock_without_redisson";@ResourceRedissonLocker redissonLocker;/*** 测试Redisson锁tryLock,不会激活看门狗,锁过期后,直接被清除* * @return 响应*/@GetMapping("/redisson/try-lock/test")public Response<String> testRedissonTryLock() {try {boolean lock1 = redissonLocker.tryLock(REDIS_LOCK_KEY, 100, 10, TimeUnit.SECONDS);if (lock1) {logger.info(">>>>>>>>得到锁");Thread.sleep(70000);return Response.success();} else {logger.info(">>>>>>>>未得到锁");return Response.fail();}} catch(Exception ex) {logger.error(">>>>>>>>Redisson异常:", ex);throw new BizException(BizExceptionCodeEnums.FAIL);} finally {logger.info(">>>>>>>释放Redisson锁");redissonLocker.unlock(REDIS_LOCK_KEY);}}/*** 测试Redisson看门狗功能:使用lock()获取锁,激活看门狗功能,每10秒自动续期30秒过期时间。* * @return 响应*/@GetMapping("/redisson/lock/test")public Response<String> testRedissonLock() {try {redissonLocker.lock(REDIS_LOCK_KEY);Thread.sleep(70000);return Response.success();} catch(Exception ex) {logger.error(">>>>>>>>Redisson异常:", ex);throw new BizException(BizExceptionCodeEnums.FAIL);} finally {logger.info(">>>>>>>释放Redisson锁");redissonLocker.unlock(REDIS_LOCK_KEY);}}
}
Redis结果:
4 结论
(1)Redisson锁存储使用Hash,Hash键为锁的键名,属性的键为UUID拼接ThreadId,格式:UUID:ThreadID,如xxx-xxx:30,属性的值为同一个线程获取锁的次数,Redisson的锁是可重入的,同一个线程获取锁,锁的数量+1,释放锁,数量减1;
(2)Redisson获取锁有两种方式,lock获取和tryLock获取,其中:
(2.1)lock获取锁时,如果第一次尝试获取获取失败,会一直等待,直到获取锁。Redisson的lock获取锁时,会为锁配置过期时间,当锁超过过期时间后,会自动释放,避免死锁,可以重新获得锁;
(2.2)lock()原始方法获取(不自定义超时(过期)时间)会激活看门狗。即通过lock()方法成功获取锁后,逻辑执行的时间超过看门狗超时时间:10秒,会自动为锁续期(增加)10秒的过期时间,使锁的过期时间保持在30秒(默认的锁过期时间);
(2.3)lock(long, TimeUnit)带参方法可以指定锁过期时间,但是,不会触发看门狗自动续期;锁过期后,自动释放锁;
(2.4)tryLock方法返回标识位,获取成功,返回true,获取失败,返回false;tryLock尝试获取锁,无法获取锁时会等待,同时,tryLock有最大的等待时间,如果获取锁的等待时间超过最大等待时间,会放弃获取锁,并返回false;tryLock不会触发看门狗,无法自动为锁续期;
(2.5)tryLock可以同时指定等待锁的最大时间,以及锁过期时间,如果等待最大时间大于锁过期时间,则锁被提前释放,重新生成锁;
(3)Redisson看门狗默认巡查周期为10秒,锁续期时间(过期时间)30秒;
(4)Redisson看门口生效的方法为:lock(),其他方法均不会触发看门狗。
相关文章:

简洁易懂:源码+实战讲解Redisson并发锁及看门狗自动续期
1 缘起 有一次同事问Redisson存储的键是否为hash? 我当时,没有看Redisson的相关源码,只知道应用, 所以没有办法回答,于是开始看看Redisson实现的源码, 顺便写了一个单机Redisson测试, 发现Redi…...

TCP 三次握手和四次挥手
✏️作者:银河罐头 📋系列专栏:JavaEE 🌲“种一棵树最好的时间是十年前,其次是现在” 目录TCP 建立连接(三次握手)为啥不能是 4 次?为啥不能是 2 次?三次握手的意义:TCP 断开连接(四…...

JavaWeb复习
JavaWeb复习一.概述1.概念2.B/S和C/S 架构二.HTTP通信协议概述1.概念2.HTTP1.0 与 HTTP1.1 版本3.HTTP 协议组成4.常见状态码5.GET 与 POST 请求方式三.Tomcat1.Web服务器介绍2.安装(Windows)3.Tomcat目录结构4.server.xml部分配置解释四.Servlet1.概念2…...

P14 PyTorch AutoGrad
前言:激活函数与loss的梯度PyTorch 提供了Auto Grad 功能,这里系统讲解一下torch.autograd.grad系统的工作原理,了解graph 结构目录:1: require_grad False2: require_grad True3: 多层bakcward 原理4: in…...

前端报表如何实现无预览打印解决方案或静默打印
在前端开发中,除了将数据呈现后,我们往往需要为用户提供,打印,导出等能力,导出是为了存档或是二次分析,而打印则因为很多单据需要打印出来作为主要的单据来进行下一环节的票据支撑, 而前端打印可…...
Operating System Course 2 - My OS
Computer Startup process上一篇:http://t.csdn.cn/XfUKt 讲到这个启动设备的第一个扇区:引导扇区。那么引导扇区的代码长什么样子?这里得看引导扇区代码源文件bootsect.s(.s后缀文件为用汇编语言编写的源代码文件)。另…...

离散数学 课时一 命题逻辑的基本概念
1 命题 1、命题:可以判断其真值的陈述句 2、真值:真或者假(1或者0) 3、真命题:真值为真的命题 4、假命题:真值为假的命题 5、原子命题:不可以再被分解成更简单的命题 6、复合命题:由原子命题通过联结词联结…...

Word文档带有权限密码怎么办?
Word文档的权限密码指的是什么?其实这是Word文档的保护方法之一,具体指Word文档的编辑、修改受到了限制,需要输入密码才能进行。 设置了权限密码的Word文档还是可以直接打开,只有当需要编辑或者修改内容的时候,才会发…...

C++多态
1. 多态的概念1.1 概念多态的概念:通俗来说,就是多种形态,具体点就是去完成某个行为,当不同的对象去完成时会产生出不同的状态举个例子:比如买票这个行为,当普通人买票时,是全价买票;…...
访问学者如何申请美国J1签证?
一、申请美国J1签证的步骤: 第一步:填写I901表。 填写I901表会收取SERVIS费用180美元,可以用VISA/Master卡直接网上支付。填完后打印收据单或者存成PDF后续再打印,记下I901收据编号。 第二步:DS-160表填写。 填写DS-…...

使用gitlab ci/cd来发布一个.net 项目
gitlab runner的安装和基本使用:https://bear-coding.blog.csdn.net/article/details/120591711安装并给项目配置完gitlab runner后再操作后面步骤。实现目标:master分支代码有变更的时候自动构建build。当开发人员在gitlab上给项目打一个tag标签分支的时候自动触发…...
笔试题-2023-蔚来-数字芯片设计【纯净题目版】
回到首页:2023 数字IC设计秋招复盘——数十家公司笔试题、面试实录 推荐内容:数字IC设计学习比较实用的资料推荐 题目背景 笔试时间:2022.08.24应聘岗位:校招-芯片逻辑综合工程师-智能硬件笔试时长:90min笔试平台:nowcoder牛客网题目类型:不定项选择题(15道)、填空题…...

ThreadLocal 详解
ThreadLocal简介JDK源码对ThreadLocal类的注释如下:ThreadLocal提供线程局部变量,使得每个线程都有自己的、独立初始化的变量副本ThreadLocal实例通常是类中的private static字段,用于将状态与线程相关联,如用户ID、事务ID只要线程…...

【Java 面试合集】重写以及重载有什么区别能简单说说嘛
重写以及重载有什么区别能简单说说嘛 前述 这是一道非常基础的面试题,我们在回答的过程中一定要逐一横向比较。 从方法的 修饰符,返回值,方法名,含义,参数等方面进行逐一分析来比较不同。 话不多话,看下…...
到底什么是股票委托接口?
在量化股票市场上,常见的股票委托接口其实有着不一样的交集,就拿股票交易接口,在量化股票跟程序化交易中,有共同之处就是在于直接委托执行下单,并且能很快的就能够将策略输出在账户持仓数据中,继续缓存下来…...
Linux驱动:VPU
1. 前言 限于作者能力水平,本文可能存在谬误,因此而给读者带来的损失,作者不做任何承诺。 2. 概述 VPU 是用来进行图像、视频数据进行硬件编、解码的硬件模块。内部集成了 Encoder、Decoder 功能部件进行图像、视频数据进行硬件编、解码&a…...

简介Servlet
目录 一、maven中心库 二、简介Servlet 三、实现Servlet动态页面 1、创建一个maven项目 2、引入依赖 3、创建目录结构 4、编写Servlet代码 5、打包 6、部署 7、验证程序 四、Servlet的运行原理 五、Tomcat伪代码 1、Tomcat初始化 a、让Tomcat先从指定的目录…...

Learning C++ No.7
引言: 北京时间:20223/2/9/22:20,距离大一下学期开学还有2天,昨天收到好消息,开学不要考试了,我并不是害怕考试,考试在我心里,地位不高,可能只有当我挂了,才能…...

【MyBatis】第八篇:一级,二级缓存
其实缓存字面的意思就是将一些内容缓存下来,等下次使用的时候可以直接调用,通过数据库得到数据,有时候会使用相同的数据,所以mybatis自然也支持缓存。 而mybatis按照缓存的效果可以分两大类:一级缓存和二级缓存。 一…...

【大唐杯备考】——5G基站开通与调测(学习笔记)
📖 前言:本期介绍5G基站开通与调测。 目录🕒 1. 概述🕒 2. 5G基站开通与调测基础🕘 2.1 3.5GHz单模100MHz配置(S111)🕘 2.2 3.5GHz单模100MHz配置(S111111)&a…...
Vim 调用外部命令学习笔记
Vim 外部命令集成完全指南 文章目录 Vim 外部命令集成完全指南核心概念理解命令语法解析语法对比 常用外部命令详解文本排序与去重文本筛选与搜索高级 grep 搜索技巧文本替换与编辑字符处理高级文本处理编程语言处理其他实用命令 范围操作示例指定行范围处理复合命令示例 实用技…...

376. Wiggle Subsequence
376. Wiggle Subsequence 代码 class Solution { public:int wiggleMaxLength(vector<int>& nums) {int n nums.size();int res 1;int prediff 0;int curdiff 0;for(int i 0;i < n-1;i){curdiff nums[i1] - nums[i];if( (prediff > 0 && curdif…...
如何为服务器生成TLS证书
TLS(Transport Layer Security)证书是确保网络通信安全的重要手段,它通过加密技术保护传输的数据不被窃听和篡改。在服务器上配置TLS证书,可以使用户通过HTTPS协议安全地访问您的网站。本文将详细介绍如何在服务器上生成一个TLS证…...
汇编常见指令
汇编常见指令 一、数据传送指令 指令功能示例说明MOV数据传送MOV EAX, 10将立即数 10 送入 EAXMOV [EBX], EAX将 EAX 值存入 EBX 指向的内存LEA加载有效地址LEA EAX, [EBX4]将 EBX4 的地址存入 EAX(不访问内存)XCHG交换数据XCHG EAX, EBX交换 EAX 和 EB…...

微软PowerBI考试 PL300-在 Power BI 中清理、转换和加载数据
微软PowerBI考试 PL300-在 Power BI 中清理、转换和加载数据 Power Query 具有大量专门帮助您清理和准备数据以供分析的功能。 您将了解如何简化复杂模型、更改数据类型、重命名对象和透视数据。 您还将了解如何分析列,以便知晓哪些列包含有价值的数据,…...

初探Service服务发现机制
1.Service简介 Service是将运行在一组Pod上的应用程序发布为网络服务的抽象方法。 主要功能:服务发现和负载均衡。 Service类型的包括ClusterIP类型、NodePort类型、LoadBalancer类型、ExternalName类型 2.Endpoints简介 Endpoints是一种Kubernetes资源…...

【免费数据】2005-2019年我国272个地级市的旅游竞争力多指标数据(33个指标)
旅游业是一个城市的重要产业构成。旅游竞争力是一个城市竞争力的重要构成部分。一个城市的旅游竞争力反映了其在旅游市场竞争中的比较优势。 今日我们分享的是2005-2019年我国272个地级市的旅游竞争力多指标数据!该数据集源自2025年4月发表于《地理学报》的论文成果…...
精益数据分析(98/126):电商转化率优化与网站性能的底层逻辑
精益数据分析(98/126):电商转化率优化与网站性能的底层逻辑 在电子商务领域,转化率与网站性能是决定商业成败的核心指标。今天,我们将深入解析不同类型电商平台的转化率基准,探讨页面加载速度对用户行为的…...
更新 Docker 容器中的某一个文件
🔄 如何更新 Docker 容器中的某一个文件 以下是几种在 Docker 中更新单个文件的常用方法,适用于不同场景。 ✅ 方法一:使用 docker cp 拷贝文件到容器中(最简单) 🧰 命令格式: docker cp <…...

篇章一 论坛系统——前置知识
目录 1.软件开发 1.1 软件的生命周期 1.2 面向对象 1.3 CS、BS架构 1.CS架构编辑 2.BS架构 1.4 软件需求 1.需求分类 2.需求获取 1.5 需求分析 1. 工作内容 1.6 面向对象分析 1.OOA的任务 2.统一建模语言UML 3. 用例模型 3.1 用例图的元素 3.2 建立用例模型 …...