18.2.go语言redis中使用lua脚本

 在 Redis 中使用 Lua 脚本可以实现原子性操作、减少网络开销以及提高执行效率。

Redis 执行 Lua 脚本的原理

Redis 内置了 Lua 解释器,能够直接在服务器端执行 Lua 脚本。当执行 Lua 脚本时,Redis 会将脚本作为一个整体执行,保证脚本执行期间不会被其他命令插入,从而实现原子性操作。

基本使用方法

 使用EVAL命令执行 Lua 脚本

EVAL命令用于在 Redis 中执行 Lua 脚本,其基本语法如下:

EVAL script numkeys key [key ...] arg [arg ...]
  • script:要执行的 Lua 脚本。
  • numkeys:脚本中使用的键名参数的数量。
  • key [key ...]:键名参数列表。
  • arg [arg ...]:其他参数列表。
redis-cli EVAL "return redis.call('SET', KEYS[1], ARGV[1])" 1 mykey myvalue
  • 脚本return redis.call('SET', KEYS[1], ARGV[1])的作用是调用 Redis 的SET命令,将ARGV[1]的值设置到KEYS[1]对应的键上。
  • 1表示脚本中使用的键名参数的数量为 1。
  • mykey是键名参数。
  • myvalue是其他参数。
使用EVALSHA命令执行预加载的脚本

为了避免每次执行脚本时都传输整个脚本内容,可以使用SCRIPT LOAD命令将脚本加载到 Redis 中,得到一个 SHA1 哈希值,然后使用EVALSHA命令通过哈希值来执行脚本。

# 加载脚本并获取SHA1哈希值
redis-cli SCRIPT LOAD "return redis.call('SET', KEYS[1], ARGV[1])"
# 输出示例:"a1b2c3d4e5f6..."# 使用EVALSHA命令执行脚本
redis-cli EVALSHA "a1b2c3d4e5f6..." 1 mykey myvalue

 在 Go 语言中使用 Lua 脚本操作 Redis

以下是一个使用 Go 语言和go-redis库执行 Lua 脚本的示例

package mainimport ("context""fmt""github.com/go-redis/redis/v8"
)func main() {rdb := redis.NewClient(&redis.Options{Addr:     "localhost:6379",Password: "",DB:       0,})ctx := context.Background()// 定义Lua脚本script := `local key = KEYS[1]local value = ARGV[1]return redis.call('SET', key, value)`// 执行Lua脚本result, err := rdb.Eval(ctx, script, []string{"mykey"}, "myvalue").Result()if err != nil {fmt.Println("Failed to execute Lua script:", err)return}fmt.Println("Script execution result:", result)
}    

Lua 脚本中的 Redis API

在 Lua 脚本中,可以使用redis.callredis.pcall函数来调用 Redis 命令:

  • redis.call:调用 Redis 命令,如果命令执行出错,脚本会终止并返回错误信息。
  • redis.pcall:调用 Redis 命令,如果命令执行出错,脚本不会终止,而是返回一个包含错误信息的 Lua 表。

call示例:

package mainimport ("context""fmt""github.com/go-redis/redis/v8"
)func main() {rdb := redis.NewClient(&redis.Options{Addr:     "localhost:6379",Password: "",DB:       0,})ctx := context.Background()// 定义Lua脚本script := `redis.call('SET', KEYS[1], ARGV[1])local value = redis.call('GET', KEYS[1])return value`// 执行Lua脚本result, err := rdb.Eval(ctx, script, []string{"mykey"}, "myvalue").Result()if err != nil {fmt.Println("Failed to execute Lua script:", err)return}fmt.Println("Script execution result:", result)
}    

pcall示例: 

package mainimport ("context""fmt""github.com/go-redis/redis/v8"
)func main() {rdb := redis.NewClient(&redis.Options{Addr:     "localhost:6379",Password: "",DB:       0,})ctx := context.Background()// 定义包含 redis.pcall 的 Lua 脚本script := `local result = redis.pcall('GET', 'nonexistent_key')if type(result) == 'table' and result.err thenreturn 'Error: ' .. result.errelsereturn resultend`// 执行 Lua 脚本result, err := rdb.Eval(ctx, script, []string{}).Result()if err != nil {fmt.Println("Failed to execute Lua script:", err)return}fmt.Println("Script execution result:", result)
}    

注意事项

  • 原子性:Lua 脚本在 Redis 中是原子执行的,但要注意脚本的执行时间不宜过长,否则会阻塞其他客户端的请求。
  • 性能:合理使用 Lua 脚本可以减少网络开销和提高执行效率,但如果脚本过于复杂,可能会影响性能。
  • 数据类型:Lua 脚本中的数据类型和 Redis 的数据类型需要进行适当的转换。

Lua 注释

单行注释

在 Lua 里,使用两个连字符 -- 开启单行注释。在 -- 之后直到该行结束的内容都会被视为注释,不会被 Lua 解释器执行。

-- 这是一个单行注释
local num = 10 -- 定义一个变量 num 并赋值为 10
多行注释

多行注释以 --[[ 开头,以 ]] 结尾。在这两个标记之间的所有内容都属于注释,无论跨多少行。

--[[
这是一个多行注释
可以包含很多行内容
用于对代码块进行详细说明
]]
local str = "Hello, World!"

Lua 基本语法

变量

Lua 是动态类型语言,变量不需要预先声明类型。常见的变量类型有 nilbooleannumberstringtablefunctionthread 和 userdata

  • 全局变量:默认情况下,变量都是全局变量。未赋值的全局变量值为 nil
-- 定义一个全局变量
message = "Hello, Lua!"
print(message)
  • 局部变量:使用 local 关键字声明局部变量,其作用域仅限于声明它的代码块。
dolocal num = 20print(num) -- 可以在代码块内访问
end
-- print(num) -- 这里会出错,因为 num 是局部变量,超出作用域
数据类型
1. 布尔类型(boolean)

只有两个值:true 和 false。在条件判断中,除了 false 和 nil 被视为假,其他值都被视为真。

local isEnabled = true
if isEnabled thenprint("Enabled")
end
2. 数字类型(number)

Lua 中的数字类型默认是双精度浮点数。

local num1 = 10
local num2 = 3.14
print(num1 + num2)
3. 字符串类型(string)

可以使用单引号或双引号来表示字符串。

local str1 = 'Hello'
local str2 = "World"
print(str1 .. " " .. str2) -- 使用 .. 进行字符串拼接
4. 表类型(table)

Lua 中最强大的数据类型,可以当作数组、字典等使用。表使用花括号 {} 来创建。

-- 当作数组使用
local fruits = {"apple", "banana", "cherry"}
print(fruits[1]) -- 索引从 1 开始-- 当作字典使用
local person = {name = "John", age = 30}
print(person.name)
控制结构
1. if-else 语句
local score = 80
if score >= 90 thenprint("A")
elseif score >= 80 thenprint("B")
elseprint("C")
end
2. for 循环
  • 数值型 for 循环:用于遍历一个数值范围。
for i = 1, 5 doprint(i)
end
  • 泛型 for 循环:用于遍历迭代器,如数组或表。
local fruits = {"apple", "banana", "cherry"}
for index, value in ipairs(fruits) doprint(index, value)
end
3. while 循环
local count = 0
while count < 3 doprint(count)count = count + 1
end

函数

函数使用 function 关键字定义,可以有参数和返回值。

function add(a, b)return a + b
endlocal result = add(3, 5)
print(result)

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

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

相关文章

⭐Unity_Demolition Media Hap (播放Hap格式视频 超16K大分辨率视频 流畅播放以及帧同步解决方案)

播放大分辨率视频以及实现局域网视频同步是许多开发者会遇到的需求,AVPro有一个 Ultra Edition版本,也能播放Hap格式视频,之外就是Demolition Media Hap插件啦,实测即使是 7208*3808 大分辨率的视频帧率还是能稳定在30帧,它能帮助我们轻松解决这些问题😎。 一、插件概述 …

AI大模型知识与医疗项目实践 - Java架构师面试实战

AI大模型知识与医疗项目实践 - Java架构师面试实战 本文模拟了一场互联网大厂的Java架构师面试&#xff0c;围绕AI大模型知识、工具以及其在医疗项目中的实践和趋势展开讨论。 第一轮提问 面试官&#xff1a; 马架构&#xff0c;请您介绍一下AI大模型的基本概念及其在医疗领…

Windows 的文件系统不区分大小写,Linux区分

在 Windows 系统中&#xff0c;文件系统默认是不区分大小写的。这意味着在 Windows 上&#xff0c;文件名 ui_BalanceMeasureScreenUI.h 和 ui_balancemeasurescreenui.h 被视为同一个文件。因此&#xff0c;即使你在代码中使用了不同的大小写方式来引用同一个文件&#xff0c;…

Unity 资源合理性检测

一&#xff1a;表格过度配置&#xff0c;表格资源是否在工程中存在&#xff0c;并输出不存在的资源 import pandas as pd import glob import osassets [] count 0# 遍历configs文件夹下所有xlsx文件 for file_path in glob.glob(configs/*.xlsx):count 1try:sheets pd.re…

Python爬虫实战:获取高考资源网各学科精品复习资料

一、引言 高考资源网拥有丰富的高考复习资料,对于我们而言,获取这些资源并整理分享能为考生提供有价值的帮助。然而,手动从网站查找和下载资源效率低且易出错。利用 Python 爬虫技术可实现自动化资源获取,提高工作效率。但在爬取过程中,需考虑网站反爬机制,采取相应措施…

DuckDB:现代数据分析的“SQLite“内核革命

在数据工程、数据科学快速演进的今天&#xff0c;一个新的名字正在快速蹿红&#xff1a;DuckDB。 有人称它是数据分析领域的SQLite&#xff0c;也有人称它为下一代轻量级OLAP引擎。 无论哪种称呼&#xff0c;都离不开一个事实&#xff1a; DuckDB 重新定义了小型数据仓库和本地…

GIS开发笔记(16)解决基于osg和osgearth三维地图上添加placeNode图标点击不易拾取的问题

一、实现效果 二、实现原理 在图标添加的位置同时添加一个红色圆球,半径为5000~8000米,图标和圆球挂接到同一个group节点,group节点再挂接到根节点,当点击到圆球时,通过遍历父节点就可以找到被点击的图标节点。 三、参考代码 //添加图标代码 #pragma once #include &…

计算机网络学习笔记 1-3章

第 1 章 计算机网络体系结构 【考纲内容】 &#xff08;一&#xff09;计算机网络概述 计算机网络的概念、组成与功能&#xff1b;计算机网络的分类&#xff1b; 计算机网络的性能指标 &#xff08;二&#xff09;计算机网络体系结构与参考模型 计算机网络分层结构&#xff…

基于NVIDIA RTX 4090的COLMAP 3.7安装指南:Ubuntu 20.04 + CUDA 11.8环境配置【2025最新版!!】

一、引言 三维重建技术作为计算机视觉领域的核心方向&#xff0c;在数字孪生、自动驾驶等领域具有重要应用价值。COLMAP作为开源的SfM&#xff08;Structure-from-Motion&#xff09;工具&#xff0c;其GPU加速特性可显著提升重建效率。由于最新研究三维重建的需要&#xff08…

Spring Boot 依赖管理: `spring-boot-starter-parent` 与 `spring-boot-dependencies`

前言 在 Spring Boot 的开发实践中&#xff0c;依赖管理是构建高质量应用的基础。spring-boot-starter-parent 和 spring-boot-dependencies 是 Spring Boot 提供的两大核心依赖管理工具&#xff0c;它们在简化依赖版本控制、统一配置等方面发挥着关键作用。 一、核心概念解析…

【MySQL】基本查询

目录 增加 查询 基本查询 where子句 结果排序 筛选分页结果 修改(更新) 删除 普通删除 截断表 插入查询结果 聚合函数 分组查询 这一节的内容是对表内容的增删查改&#xff0c;其中重点是表的查询 增加 语法&#xff1a; INSERT [INTO] table_name [(column [, …

【C++详解】C++入门(二)引用、内联函数、nullptr宏

文章目录 一、引用引用的概念和定义引用的功能引用的特性const引用const用法回顾权限的放大缩小const引用的功能 指针和引用的关系 二、内联函数三、nullptr补充结构体指针变量类型重定义 一、引用 引用的概念和定义 C祖师爷为了优化在部分场景中使用指针会出现的效率较低和比…

毕业设计-基于深度学习的实时网络入侵检测系统

项目技术说明 深度学习实时网络入侵检测系统是一种利用深度学习技术对网络流量进行实时分析&#xff0c;以识别和阻止潜在网络攻击的安全解决方案。相比传统基于规则的入侵检测系统(IDS)&#xff0c;这种系统能够通过学习网络流量的正常模式和异常模式&#xff0c;更有效地检测…

中药企业数字化转型:从传统制造到智能制药的跨越

在当今数字化浪潮下&#xff0c;中药企业正积极拥抱变革&#xff0c;努力实现从传统制造向智能制药的跨越&#xff0c;以适应市场竞争和满足人们对中药质量与效率的更高要求。 在原料管理环节&#xff0c;企业通过采用物联网技术&#xff0c;对中药材种植、采集过程进行全程监…

Vue 2 的响应式 API 和 Vue 3 的组合式 API 的详细对比,从核心机制、使用方式、代码示例及优缺点展开

以下是 Vue 2 的响应式 API 和 Vue 3 的组合式 API 的详细对比&#xff0c;从核心机制、使用方式、代码示例及优缺点展开&#xff1a; 1. Vue 2 的响应式 API 核心机制 基于 Object.defineProperty&#xff1a; 通过劫持对象的 getter 和 setter 实现数据变化追踪。限制&…

“八股训练营”学习总结

在参加为期 40 天的八股训练营的这段时间里&#xff0c;我收获满满&#xff0c;不仅在知识技能上得到了提升&#xff0c;更在学习习惯和自我认知方面有了很大的进步。 在知识层面&#xff0c;训练营涵盖了网络、数据库、缓存以及python测试开发等多方面的知识点。 网络方面&a…

Python对比两张CAD图并标记差异的解决方案

以下是使用Python对比两张CAD图并标记差异的解决方案&#xff0c;结合图像处理和CAD结构分析&#xff1a; 一、环境准备与库选择 图像处理库&#xff1a;使用OpenCV进行图像差异检测、颜色空间转换和轮廓分析。CAD解析库&#xff1a;若为DXF格式&#xff0c;使用ezdxf解析实体…

记录学习记录学习《手动学习深度学习》这本书的笔记(九)

马不停蹄地来到了第十二章&#xff1a;计算性能…… 感觉应该是讲并行计算方面的&#xff0c;比如GPU、CPU、CUDA那些。 第十二章&#xff1a;计算性能 12.1 编译器和解释器 这里先提出了命令式编程和符号式编程的概念。 命令式编程VS符号式编程 目前为止&#xff0c;本书…

模板引擎语法-过滤器

模板引擎语法-过滤器 文章目录 模板引擎语法-过滤器[toc]1.default过滤器2.default_if_none过滤器3.length过滤器4.addslashes过滤器5.capfirst过滤器6.cut过滤器7.date过滤器8.dictsort过滤器 1.default过滤器 default过滤器用于设置默认值。default过滤器对于变量的作用&…

make学习三:书写规则

系列文章目录 Make学习一&#xff1a;make初探 Make学习二&#xff1a;makefile组成要素 文章目录 系列文章目录前言默认目标规则语法order-only prerequisites文件名中的通配符伪目标 Phony Targets没有 Prerequisites 和 recipe内建特殊目标名一个目标多条规则或多个目标共…