lua是一种轻量小巧的脚本语言,用标准C语言编写并以源代码形式开放, 其设计目的是为了嵌入应用程序中,从而为应用程序提供灵活的扩展和定制功能。
Lua应用场景:游戏开发、独立应用脚本、Web应用脚本、扩展和数据库插件。 OpenRestry:一个可伸缩的基于Nginx的Web平台,是在nginx之上集成了lua模块的第三方服务器 OpenResty是一个通过Lua扩展Nginx实现的可伸缩的Web平台,内部集成了大量精良的Lua库、第三方模块以及大多数的依赖项。 用于方便地搭建能够处理超高并发(日活千万级别)、扩展性极高的动态Web应用、Web服务和动态网关。 功能和nginx类似,就是由于支持lua动态脚本,所以更加灵活,可以实现鉴权、限流、分流、日志记录、灰度发布等功能。 OpenResty通过Lua脚本扩展nginx功能,可提供负载均衡、请求路由、安全认证、服务鉴权、流量控制与日志监控等服务。 类似的还有Kong(Api Gateway)、tengine(阿里)
如果报错,说找不到readline/readline.h, 可以通过yum命令安装
yum -y install readline-devel ncurses-devel安装完以后再
make linux / make install最后,直接输入 lua命令即可进入lua的控制台. Lua环境协作组件 从Redis2.6.0版本开始,通过内置的lua编译/解释器,可以使用EVAL命令对lua脚本进行求值。 脚本的命令是原子的,RedisServer在执行脚本命令中,不允许插入新的命令 脚本的命令可以复制,RedisServer在获得脚本后不执行,生成标识返回,Client根据标识就可以随时执行
EVAL命令 通过执行redis的eval命令,可以运行一段lua脚本
EVAL script numkeys key [key ...] arg [arg ...]命令说明: script参数:是一段Lua脚本程序,它会被运行在Redis服务器上下文中,这段脚本不必(也不应该) 定义为一个Lua函数。 numkeys参数:用于指定键名参数的个数。 key [key …]参数: 从EVAL的第三个参数开始算起,使用了numkeys个键(key),表示在脚本中 所用到的那些Redis键(key),这些键名参数可以在Lua中通过全局变量KEYS数组,用1为基址的形 式访问( KEYS[1] , KEYS[2] ,以此类推)。 arg [arg …]参数:可以在Lua中通过全局变量ARGV数组访问,访问的形式和KEYS变量类似( ARGV[1] 、 ARGV[2] ,诸如此类)
eval "return {KEYS[1],KEYS[2],ARGV[1],ARGV[2]}" 2 key1 key2 first secondlua脚本中调用Redis命令
redis.call(): 返回值就是redis命令执行的返回值 如果出错,则返回错误信息,不继续执行redis.pcall(): 返回值就是redis命令执行的返回值 如果出错,则记录错误信息,继续执行注意事项 在脚本中,使用return语句将返回值返回给客户端,如果没有return,则返回nil eval "return redis.call('set',KEYS[1],ARGV[1])" 1 n1 zhaoyunEVAL 命令要求你在每次执行脚本的时候都发送一次脚本主体(script body)。 Redis 有一个内部的缓存机制,因此它不会每次都重新编译脚本,不过在很多场合,付出无谓的带宽来 传送脚本主体并不是最佳选择。 为了减少带宽的消耗, Redis 实现了 EVALSHA 命令,它的作用和 EVAL 一样,都用于对脚本求值,但 它接受的第一个参数不是脚本,而是脚本的 SHA1 校验和(sum)
SCRIPT FLUSH :清除所有脚本缓存 SCRIPT EXISTS :根据给定的脚本校验和,检查指定的脚本是否存在于脚本缓存 SCRIPT LOAD :将一个脚本装入脚本缓存,返回SHA1摘要,但并不立即运行它
192.168.24.131:6380> script load "return redis.call('set',KEYS[1],ARGV[1])" "c686f316aaf1eb01d5a4de1b0b63cd233010e63d" 192.168.24.131:6380> evalsha c686f316aaf1eb01d5a4de1b0b63cd233010e63d 1 n2 zhangfei OK 192.168.24.131:6380> get n2SCRIPT KILL :杀死当前正在运行的脚本
使用redis-cli直接执行lua脚本。 test.lua
return redis.call('set',KEYS[1],ARGV[1]) ./redis-cli -h 127.0.0.1 -p 6379 --eval test.lua name:6 , 'caocao' #,两边有空格list.lua
local key=KEYS[1] local list=redis.call("lrange",key,0,-1); return list; ./redis-cli --eval list.lua list利用Redis整合Lua,主要是为了性能以及事务的原子性。因为redis帮我们提供的事务功能太差。
Redis 传播 Lua 脚本,在使用主从模式和开启AOF持久化的前提下: 当执行lua脚本时,Redis 服务器有两种模式:脚本传播模式和命令传播模式。 脚本传播模式 脚本传播模式是 Redis 复制脚本时默认使用的模式 Redis会将被执行的脚本及其参数复制到 AOF 文件以及从服务器里面。 执行以下命令:
eval "redis.call('set',KEYS[1],ARGV[1]);redis.call('set',KEYS[2],ARGV[2])" 2 n1 n2 zhaoyun1 zhaoyun2那么主服务器将向从服务器发送完全相同的 eval 命令
eval "redis.call('set',KEYS[1],ARGV[1]);redis.call('set',KEYS[2],ARGV[2])" 2 n1 n2 zhaoyun1 zhaoyun2注意:在这一模式下执行的脚本不能有时间、内部状态、随机函数(spop)等。执行相同的脚本以及参数 必须产生相同的效果。在Redis5,也是处于同一个事务中
处于命令传播模式的主服务器会将执行脚本产生的所有写命令用事务包裹起来,然后将事务复制到 AOF 文件以及从服务器里面。 因为命令传播模式复制的是写命令而不是脚本本身,所以即使脚本本身包含时间、内部状态、随机函数 等,主服务器给所有从服务器复制的写命令仍然是相同的。 为了开启命令传播模式,用户在使用脚本执行任何写操作之前,需要先在脚本里面调用以下函数:
redis.replicate_commands()
redis.replicate_commands() 只对调用该函数的脚本有效:在使用命令传播模式执行完当前脚本之后, 服务器将自动切换回默认的脚本传播模式。 如果我们在主服务器执行以下命令:
eval "redis.replicate_commands();redis.call('set',KEYS[1],ARGV[1]);redis.call('set',K EYS[2],ARGV[2])" 2 n1 n2 zhaoyun11 zhaoyun22那么主服务器将向从服务器复制以下命令
EXEC *1 $5 MULTI *3 $3 set $2 n1 $9 zhaoyun11 *3 $3 set $2 n2 $9 zhaoyun22 *1 $4 EXEC 管道(pipeline),事务和脚本(lua)三者的区别 三者都可以批量执行命令 管道无原子性,命令都是独立的,属于无状态的操作 事务和脚本是有原子性的,其区别在于脚本可借助Lua语言可在服务器端存储的便利性定制和简化操作脚本的原子性要强于事务,脚本执行期间,另外的客户端 其它任何脚本或者命令都无法执行,脚本的执行时间应该尽量短,不能太耗时的脚本我们都知道MySQL有慢查询日志 Redis也有慢查询日志,可用于监视和优化查询 慢查询设置 在redis.conf中可以配置和慢查询日志相关的选项
#执行时间超过多少微秒的命令请求会被记录到日志上 0 :全记录 <0 不记录 slowlog-log-slower-than 10000 #slowlog-max-len 存储慢查询日志条数 slowlog-max-len 128Redis使用列表存储慢查询日志,采用队列方式(FIFO) config set的方式可以临时设置,redis重启后就无效 config set slowlog-log-slower-than 微秒 config set slowlog-max-len 条数 查看日志:slowlog get [n]
127.0.0.1:6379> config set slowlog-log-slower-than 0 OK 127.0.0.1:6379> config set slowlog-max-len 2 OK 127.0.0.1:6379> set name:001 zhaoyun OK 127.0.0.1:6379> set name:002 zhangfei OK 127.0.0.1:6379> get name:002 "zhangfei"127.0.0.1:6379> slowlog get
(integer) 7 #日志的唯一标识符(uid) (integer) 1589774302 #命令执行时的UNIX时间戳(integer) 65 #命令执行的时长(微秒) “get” #执行命令及参数 “name:002”“127.0.0.1:37277”“” (integer) 6 (integer) 1589774281(integer) 7 “set” “name:002”“zhangfei”“127.0.0.1:37277”“” set和get都记录,第一条被移除了在redisServer中保存和慢查询日志相关的信息
struct redisServer { // ... // 下一条慢查询日志的 ID long long slowlog_entry_id; // 保存了所有慢查询日志的链表 FIFO list *slowlog; // 服务器配置 slowlog-log-slower-than 选项的值 long long slowlog_log_slower_than; // 服务器配置 slowlog-max-len 选项的值 unsigned long slowlog_max_len; // ... };lowlog 链表保存了服务器中的所有慢查询日志, 链表中的每个节点都保存了一个 slowlogEntry 结 构, 每个 slowlogEntry 结构代表一条慢查询日志。
typedef struct slowlogEntry { // 唯一标识符 long long id; // 命令执行时的时间,格式为 UNIX 时间戳 time_t time; // 执行命令消耗的时间,以微秒为单位 long long duration; // 命令与命令参数 robj **argv; // 命令与命令参数的数量 int argc; } slowlogEntry;在每次执行命令的之前和之后, 程序都会记录微秒格式的当前 UNIX 时间戳, 这两个时间戳之间的差就是服务器执行命令所耗费的时长, 服务器会将这个时长作为参数之一传给slowlogPushEntryIfNeeded 函数, 而 slowlogPushEntryIfNeeded 函数则负责检查是否需要为这次执行的命令创建慢查询日志
// 记录执行命令前的时间 before = unixtime_now_in_us() //执行命令 execute_command(argv, argc, client) //记录执行命令后的时间 after = unixtime_now_in_us() // 检查是否需要创建新的慢查询日志 slowlogPushEntryIfNeeded(argv, argc, before-after) void slowlogPushEntryIfNeeded(robj **argv, int argc, long long duration) { if (server.slowlog_log_slower_than < 0) return; /* Slowlog disabled */ /* 负 数表示禁用 */ if (duration >= server.slowlog_log_slower_than) /* 如果执行时间 > 指定阈值*/ listAddNodeHead(server.slowlog,slowlogCreateEntry(argv,argc,duration)); /* 创建一个slowlogEntry对象,添加到列表首部*/ while (listLength(server.slowlog) > server.slowlog_max_len) /* 如果列表长度 > 指定长度 */ listDelNode(server.slowlog,listLast(server.slowlog)); /* 移除列表尾部元素 */ }slowlogPushEntryIfNeeded 函数的作用有两个:
检查命令的执行时长是否超过 slowlog-log-slower-than 选项所设置的时间, 如果是的话, 就 为命令创建一个新的日志, 并将新日志添加到 slowlog 链表的表头。检查慢查询日志的长度是否超过 slowlog-max-len 选项所设置的长度, 如果是的话, 那么将多 出来的日志从 slowlog 链表中删除掉使用slowlog get 可以获得执行较慢的redis命令,针对该命令可以进行优化: 1、尽量使用短的key,对于value有些也可精简,能使用int就int。 2、避免使用keys *、hgetall等全量操作。 3、减少大key的存取,打散为小key 4、将rdb改为aof模式 rdb fork 子进程 主进程阻塞 redis大幅下降 关闭持久化 , (适合于数据量较小) 改aof 命令式 5、想要一次添加多条数据的时候可以使用管道 6、尽可能地使用哈希存储 7、尽量限制下redis使用的内存大小,这样可以避免redis使用swap分区或者出现OOM错误 内存与硬盘的swap
Redis客户端通过执行MONITOR命令可以将自己变为一个监视器,实时地接受并打印出服务器当前处理 的命令请求的相关信息。 此时,当其他客户端向服务器发送一条命令请求时,服务器除了会处理这条命令请求之外,还会将这条 命令请求的信息发送给所有监视器
Redis客户端1 127.0.0.1:6379> monitor OK 1589706136.030138 [0 127.0.0.1:42907] "COMMAND" 1589706145.763523 [0 127.0.0.1:42907] "set" "name:10" "zhaoyun" 1589706163.756312 [0 127.0.0.1:42907] "get" "name:10" Redis客户端2 127.0.0.1:6379> 127.0.0.1:6379> set name:10 zhaoyun OK 127.0.0.1:6379> get name:10 "zhaoyun"redisServer 维护一个 monitors 的链表,记录自己的监视器,每次收到 MONITOR 命令之后,将客户 端追加到链表尾
void monitorCommand(redisClient *c) { /* ignore MONITOR if already slave or in monitor mode */ if (c->flags & REDIS_SLAVE) return; c->flags |= (REDIS_SLAVE|REDIS_MONITOR); listAddNodeTail(server.monitors,c); addReply(c,shared.ok); //回复OK }利用call函数实现向监视器发送命令
// call() 函数是执行命令的核心函数,这里只看监视器部分 /*src/redis.c/call*/ /* Call() is the core of Redis execution of a command */ void call(redisClient *c, int flags) { long long dirty, start = ustime(), duration; int client_old_flags = c->flags; /* Sent the command to clients in MONITOR mode, only if the commands are * not generated from reading an AOF. */ if (listLength(server.monitors) && !server.loading && !(c->cmd->flags & REDIS_CMD_SKIP_MONITOR)) { r eplicationFeedMonitors(c,server.monitors,c->db->id,c->argv,c->argc); } . ..... }call 主要调用了 replicationFeedMonitors ,这个函数的作用就是将命令打包为协议,发送给监视 器
grafana、prometheus以及redis_exporter。 Grafana 是一个开箱即用的可视化工具,具有功能齐全的度量仪表盘和图形编辑器,有灵活丰富的图形化选项,可以混合多种风格,支持多个数据源特点。 Prometheus是一个开源的服务监控系统,它通过HTTP协议从远程的机器收集数据并存储在本地的时序数据库上。 redis_exporter为Prometheus提供了redis指标的导出,配合Prometheus以及grafana进行可视化及监控。