Redis监控步骤get!Google精髓的四大法则直接掌握

本文涉及的产品
Redis 开源版,标准版 2GB
推荐场景:
搭建游戏排行榜
云数据库 Tair(兼容Redis),内存型 2GB
简介: Redis也是对外服务,所以Google四个黄金指标同样适用,还从延迟、流量、错误、饱和度分析Redis关键指标。

Redis也是对外服务,所以Google四个黄金指标同样适用,还从延迟、流量、错误、饱和度分析Redis关键指标。


1 延迟

选择Redis是想得到更快响应速度和更高吞吐量,所以延迟数据对使用Redis的应用程序至关重要。


1.1 如何监控延迟

① 客户端应用程序埋点

Java程序调用Redis时,计算各命令花费多久,然后把耗时数据推给监控系统。


灵活,想按什么维度统计就按啥


缺点:代码侵入性,和客户端埋点监控MySQL的原理一样。


② redis-cli的–latency

客户端连上 redis-server,然后不断发 ping 命令,统计耗时。我在远端机器对某 redis-server 做探测,看探测结果:


javaedge@JavaEdgedeMac-mini ~ % redis-cli --latency -h x.x.x.x -p 6379

min: 5, max: 187, avg: 8.19 (469 samples)^C

javaedge@JavaEdgedeMac-mini ~ %


跑到 localhost 机器对本机的 redis-server 做探测:


javaedge@JavaEdgedeMac-mini ~ % redis-cli --latency -h localhost -p 6379

min: 0, max: 12, avg: 0.23 (1505 samples)^C

javaedge@JavaEdgedeMac-mini ~ %


远端机器平均延迟是 8.19 ms,本地探测平均延迟是 0.23 ms,相差巨大,时间主要花在网络 I/O,Redis本身执行效率是很高的。


只使用 ping 对 redis-server 探测,怎能反映真实工作负载?


是不是ping命令执行很快,而应用发起的真实请求其实很慢?

Redis 是单线程顺序执行,若某请求执行得慢,其他所有客户端都得等,所以 ping 对 redis-server 探测,理论上探测结果就能反映 redis-server 真实情况。


若发现Redis变慢,咋找那些执行慢的命令?

slowlog,先定义执行时间超过多久算慢,Redis默认配置10ms,比如调成5ms。


javaedge@JavaEdgedeMac-mini redis % grep slower redis.conf  

# In many cases the disk is slower than the network, and storing and loading

slowlog-log-slower-than 10000

javaedge@JavaEdgedeMac-mini redis % redis-cli

127.0.0.1:6379> config set slowlog-log-slower-than 5000

OK

127.0.0.1:6379> get slowlog-log-slower-than

(nil)

127.0.0.1:6379> config get slowlog-log-slower-than

1) "slowlog-log-slower-than"

2) "5000"

127.0.0.1:6379> config rewrite

OK

127.0.0.1:6379> quit

javaedge@JavaEdgedeMac-mini redis %  grep slower redis.conf

slowlog-log-slower-than 5000



之后一些执行时间超过5ms的命令就会被记录,然后 slowlog get [count] 查看 count 出的slowlog条数。


# 获取 2 条作为示例

127.0.0.1:6379> SLOWLOG get 2

# 序号

1) 1) (integer) 47

 # 时间戳

  2) (integer) 1668743666

  # 执行时间(单位是微秒)

  3) (integer) 13168

  # 命令及其参数

  4) 1) "hset"

     2) "/idents/Default"

     3) "tt-fc-dev01.nj"

     4) "1668743666"

  # 客户端IP和端口  

  5) "127.0.0.1:43172"

  6) ""

2) 1) (integer) 46

  2) (integer) 1668646906

  3) (integer) 13873

  4) 1) "hset"

     2) "/idents/Default"

     3) "10.206.16.3"

     4) "1668646906"

  5) "127.0.0.1:44612"

  6) ""


2 流量

Redis每秒处理多少请求,每秒接收、返回多少字节,在Redis都内置相关指标,通过 redis-cli 连上Redis,执行 info all 。绝大部分监控系统都是从 info 返回内容提取的指标。


javaedge@JavaEdgedeMac-mini ~ % redis-cli -h 127.0.0.1 -p 6379 info all | grep instantaneous

# 每秒执行多少次操作

instantaneous_ops_per_sec:0

# 每秒接收多少 KiB

instantaneous_input_kbps:0.00

# 每秒返回多少 KiB

instantaneous_output_kbps:0.00

instantaneous_input_repl_kbps:0.00

instantaneous_output_repl_kbps:0.00

javaedge@JavaEdgedeMac-mini ~ %


一般一个 Redis 实例每s处理几w个请求都正常。


每s处理的操作如较恒定,则很健康。如发现 ops_per_sec 变少,注意可能:


某耗时操作导致命令阻塞


客户端出问题,不发请求过来了


若把 Redis 做缓存,还需关注指标:


javaedge@JavaEdgedeMac-mini ~ % redis-cli -h 127.0.0.1 -p 6379 info all | grep keyspace

keyspace_hits:0

keyspace_misses:1


都 Counter 型,即 Redis 实例启动以来统计的所有命中的数量和未命中数量。如统计总体命中率,使用 hits 除以总量。


hit rate = keyspace_hits / (keyspace_hits + keyspace_misses)


近期命中率,如最近10min,通过 PromQL increase 函数做二次运算:


increase(keyspace_hits[10m])

/

(increase(keyspace_hits[10m]) + increase(keyspace_misses[10m]))


命中率低于 0.8 就要注意可能:


内存不够用,很多 Key 被清了

数据没及时填充或过期了

较低命中率影响应用程序延迟,因为通常当应用程序无法从 Redis 获取缓存数据时,就要穿透到更慢的存储介质取数。


3 错误

Redis响应客户端请求时,只是操作内存,依赖较少,异常概率很小。如客户端操作 Redis 返回错误,大概率网络问题或命令写错。最好做客户端埋点监控,自己发现自己解决。


Redis 对客户端数量限制默认10w,如超过,rejected_connections 指标+1。和MySQL不同,Redis应该很少遇到超过最大连接数(maxclients) 。


4 饱和度

Redis重度使用内存,内存的使用率、碎片率及因为内存不够用而清理的Key数量都需关注。 info memory 查看:


# Memory

used_memory:26276368

used_memory_human:25.06M

used_memory_rss:39575552

used_memory_rss_human:37.74M

used_memory_peak:33979824

used_memory_peak_human:32.41M

...

maxmemory:0

maxmemory_human:0B

...

mem_fragmentation_ratio:1.51

...


used_memory 顾名思义就是使用了多少内存,是从 Redis 自身视角来看的,human后缀表示人类易读方式来展示结果。


used_memory_rss:从os视角看分配多少内存给Redis

used_memory_rss 除以 used_memory 就是内存碎片率(mem_fragmentation_ratio):

used_memory_rss(39575552) / used_memory(26276368) = mem_fragmentation_ratio(1.51)


碎片化率较高说明 used_memory_rss 相对大,used_memory 相对小,即OS给Redis分配很多内存,但Redis利用不好。OS分配内存时因为有分配单位(如8byte、16byte、32byte、64byte)限制,如Redis申请222个字节,os直接分配256个字节,就会产生内存碎片,但这种内存碎片不多,要不然OS会浪费很多内存,所以多给一些内存正常,即 mem_fragmentation_ratio稍大于1没问题。


若执行 Flushdb,会让Redis把数据删掉,而Redis不会立马把这部分内存归还给OS,碎片化率指标就会巨高,此时推荐重启Redis。毕竟,都已Flushdb,说明 Redis 里已无数据。


随应用程序不断删除、修改Redis数据,内存碎片化率也会上升,mem_fragmentation_ratio>1.5,就说明碎片率太高,需重启 Redis或命令Redis清理碎片:


CONFIG SET activedefrag yes


搞得很紧张啊,这大可不必,就像这例,RSS才占37M,Redis实际使用25M,碎片12M,很小啊,虽 mem_fragmentation_ratio已1.51 ,但无需处理。used_memory_peak 32.41M,说明曾经用到过这么多,后来有些 Key 过期或删除/修改,才导致碎片率高点。


何时需处理碎片?

机器内存不够用

碎片化内存浪费太多

且碎片化率很高时

才需处理。调整如下参数控制碎片处理过程。


# 开启自动内存碎片整理(总开关)

activedefrag yes

# 当碎片达到 100mb 时,开启内存碎片整理

active-defrag-ignore-bytes 100mb

# 当碎片超过 10% 时,开启内存碎片整理

active-defrag-threshold-lower 10

# 内存碎片超过 100%,则尽最大努力整理

active-defrag-threshold-upper 100

# 内存自动整理占用资源最小百分比

active-defrag-cycle-min 25

# 内存自动整理占用资源最大百分比

active-defrag-cycle-max 75


上面讨论 mem_fragmentation_ratio 过大case,实际上这值还可能小于1,表示Redis使用了超过RSS数量的内存,说明此时部分内存已被放到交换分区,而磁盘性能相比内存差了约5个数量级,所以出现这种情况严重影响性能。


饱和度的度量还有指标evicted_keys:当内存占用超maxmemory时,Redis清理的Key的数量。内存达maxmemory时的处理策略可配置,默认noeviction。


127.0.0.1:6379> config get maxmemory-policy

1) "maxmemory-policy"

2) "noeviction"


这些指标出问题,上游服务大概率受影响,还有一些指标虽然短期不会影响上游服务,但是如果不及时处理未来也会出现大麻烦,这类指标通常用于衡量Redis内部的一些运行状况,如:


持久化相关指标:rdb_changes_since_last_save 自从上次落盘以来又有多少次变更

主从相关指标:master_link_down_since_seconds 主从连接已断开的时长

5 采集配置

Categraf 也提供 Redis 采集插件,配置样例在 conf/input.redis/redis.toml:


[[instances]]


# 最核心配置,即Redis的连接地址

# address = "127.0.0.1:6379"

# 认证信息,username 字段低版本的 Redis 是不需要的,6.0以上版本并且启用了ACL的才需要。

# username = ""

# password = ""

# pool_size = 2

# # Optional. Specify redis commands to retrieve values

# 自定义一些命令来获取指标,类似MySQL采集器的queries ,适用于业务指标采集的场景

# commands = [

#     {command = ["get", "sample-key1"], metric = "custom_metric_name1"},

#     {command = ["get", "sample-key2"], metric = "custom_metric_name2"}

# ]

# 通用配置,所有的 Categraf 的采集器,都支持在 `[[instances]]` 下面自定义标签

# labels = { instance="n9e-dev-redis" }



个人习惯机器名过滤,便于把 Redis 指标和 Redis 所在机器的指标放到一张大盘。 仪表盘样例:

9.png



Redis 监控的原理、采集方法、仪表盘相关的知识就讲解完了,下面我们做一个总结。


6 总结

Redis也是对外服务,所以还 Google四个黄金指标的法则来梳理指标。


延迟方面使用 redis-cli --latency 探测,不过采集器一般不会直接调用这命令行工具,而是采集时先 ping 获取延迟。



流量,关注每s处理多少command,每秒收到多少网络流量,返回多少网络流量

Redis只是操作内存,基本遇不到错误,可使用客户端埋点采集网络、命令错误

饱和度方面,则重点关注内存饱和度,尤其内存碎片率,<1不行,太大也不好


8.png

7 FAQ

Redis监控关键指标和告警规则可根据具体业务需求而变化,以下是一些常见的Redis告警规则PromQL表达式:

7.png

监控Redis内存使用情况:当连续3次采集到的Redis使用的内存超过阈值时,发出告警。


ALARM: expr: REDIS_MEMORY_USAGE > $REDIS_THRESHOLD_FOR_MEMORY_USAGE

FOR: 3m

ANOMALY_DETECTION_CONFIG:

pattern: 7/10 sudden increase

lowerBound: 0

upperBound: 2


监控Redis连接数:当Redis客户端连接数超过阈值时触发告警。


redisconnectedclients > $REDIS_CLIENTS_MAX_ALERT_VALUE


监控Redis响应时间:如果Redis响应时间超过设定值,则触发告警。


redis_latency_milliseconds > $REDIS_LATENCY_THRESHOLD_VALUE


相关实践学习
基于Redis实现在线游戏积分排行榜
本场景将介绍如何基于Redis数据库实现在线游戏中的游戏玩家积分排行榜功能。
云数据库 Redis 版使用教程
云数据库Redis版是兼容Redis协议标准的、提供持久化的内存数据库服务,基于高可靠双机热备架构及可无缝扩展的集群架构,满足高读写性能场景及容量需弹性变配的业务需求。 产品详情:https://www.aliyun.com/product/kvstore &nbsp; &nbsp; ------------------------------------------------------------------------- 阿里云数据库体验:数据库上云实战 开发者云会免费提供一台带自建MySQL的源数据库&nbsp;ECS 实例和一台目标数据库&nbsp;RDS实例。跟着指引,您可以一步步实现将ECS自建数据库迁移到目标数据库RDS。 点击下方链接,领取免费ECS&amp;RDS资源,30分钟完成数据库上云实战!https://developer.aliyun.com/adc/scenario/51eefbd1894e42f6bb9acacadd3f9121?spm=a2c6h.13788135.J_3257954370.9.4ba85f24utseFl
目录
相关文章
|
2月前
|
NoSQL 关系型数据库 MySQL
2024Mysql And Redis基础与进阶操作系列(4-2)作者——LJS[含MySQL非空、唯一性、PRIMARY KEY、自增列/自增约束举例说明等详解步骤及常见报错问题对应的解决方法]
24MySQL非空、唯一性、PRIMARY KEY、自增列/自增约束举例说明等详解步骤及常见报错问题对应的解决方法(4-2) 学不会你来砍我!!!
|
22天前
|
关系型数据库 MySQL 应用服务中间件
《docker基础篇:8.Docker常规安装简介》包括:docker常规安装总体步骤、安装tomcat、安装mysql、安装redis
《docker基础篇:8.Docker常规安装简介》包括:docker常规安装总体步骤、安装tomcat、安装mysql、安装redis
80 7
|
2月前
|
NoSQL Linux PHP
如何在不同操作系统上安装 Redis 服务器,包括 Linux 和 Windows 的具体步骤
本文介绍了如何在不同操作系统上安装 Redis 服务器,包括 Linux 和 Windows 的具体步骤。接着,对比了两种常用的 PHP Redis 客户端扩展:PhpRedis 和 Predis,详细说明了它们的安装方法及优缺点。最后,提供了使用 PhpRedis 和 Predis 在 PHP 中连接 Redis 服务器及进行字符串、列表、集合和哈希等数据类型的基本操作示例。
104 4
|
2月前
|
NoSQL 安全 关系型数据库
2024Mysql And Redis基础与进阶操作系列(6)作者——LJS[含MySQL 多表之一对一/多;多对多;多表联合查询等详解步骤及常见报错问题所对应的解决方法]
MySQL 多表之一对一/多;多对多;多表联合之交叉连接;内连接;左、右、外、满、连接;子查询及关键字;自连接查询等详解步骤及常见报错问题所对应的解决方法
|
2月前
|
SQL NoSQL 关系型数据库
2024Mysql And Redis基础与进阶操作系列(5)作者——LJS[含MySQL DQL基本查询:select;简单、排序、分组、聚合、分组、分页等详解步骤及常见报错问题所对应的解决方法]
MySQL DQL基本查询:select;简单、排序、分组、聚合、分组、分页、INSERT INTO SELECT / FROM查询结合精例等详解步骤及常见报错问题所对应的解决方法
|
2月前
|
SQL NoSQL 关系型数据库
|
2月前
|
存储 SQL 关系型数据库
2024Mysql And Redis基础与进阶操作系列(1)作者——LJS[含MySQL的下载、安装、配置详解步骤及报错对应解决方法]
Mysql And Redis基础与进阶操作系列(1)之[MySQL的下载、安装、配置详解步骤及报错对应解决方法]
|
2月前
|
SQL 关系型数据库 MySQL
|
2月前
|
存储 SQL NoSQL
|
2月前
|
NoSQL 关系型数据库 MySQL
2024Mysql And Redis基础与进阶操作系列(8)作者——LJS[含MySQL 创建、修改、跟新、重命名、删除视图等具体详步骤;注意点及常见报错问题所对应的解决方法]
MySQL 创建、修改、跟新、重命名、删除视图等具体详步骤;举例说明注意点及常见报错问题所对应的解决方法