1.并发问题
redis官方生成可以达到 10万/每秒,每秒执行10万条命令
假如: 业务需要每秒100万的命令执行呢?
2.数据量太大
一台服务器内存正常是16~256G,假如你的业务需要500G内存,
新浪微博作为世界上最大的redis存储,就超过1TB的数据,去哪买这么大的内存条?各大公司有自己的解决方案,推出各自的集群功能,核心思想都是将数据分片(sharding)存储在多个redis实例中,每一片就是一个redis实例
各大企业集群方案:
twemproxy由Twitter开源
Codis由豌豆荚开发,基于GO和C开发
redis-cluster官方3.0版本后的集群方案
解决方案:
1、配置容量很高的服务器,比如CPU和内存等
2、正确的应该是考虑分布式,加机器把数据分到不同的位置,分摊集中式的压力,然后一堆机器做一件事
数据分布原理图
image.png分布式数据库首要解决把整个数据集按照分区规则映射到多个节点的问题,即把数据集划分到多个节点上,每个节点负责整个数据的一个子集。
常见的分区规则有哈希分区和顺序分区。Redis Cluster采用哈希分区规则,因此接下来会讨论哈希分区规则。
- 节点取余分区
- 一致性哈希分区
- 虚拟槽分区(redis-cluster采用的方式)
顺序分区
image.png哈希分区(节点取余)
image.png例如:按照节点取余的方式,分三个节点
1~100的数据对3取余,可以分为三类
- 余数为0
- 余数为1
- 余数为2
那么同样的分4个节点就是hash(key)%4
节点取余的优点是简单,客户端分片直接是哈希+取余
一致性哈希
Redis Cluster采用虚拟槽分区
虚拟槽分区巧妙地使用了哈希空间,使用分散度良好的哈希函数把所有的数据映射到一个固定范围内的整数集合,整数定义为槽(slot)
Redis Cluster槽的范围是0 ~ 16383
槽是集群内数据管理和迁移的基本单位。采用大范围的槽的主要目的是为了方便数据的拆分和集群的扩展,
每个节点负责一定数量的槽。
image.png
搭建Redis Cluster
搭建集群分为几部
- 准备节点(几匹马儿)
- 节点通信(几匹马儿分配主从)
- 分配槽位给节点(slot分配给马儿)
redis-cluster集群架构
多个服务端,负责读写,彼此通信,redis指定了16384个槽。
多匹马儿,负责运输数据,马儿分配16384个槽位,管理数据。
ruby的脚本自动就把分配槽位这事做了
image.png
安装方式
官方提供通过ruby语言的脚本一键安装
1.环境准备
通过配置,开启redis-cluster
port 7000
daemonize yes
dir "/opt/redis/data"
logfile "7000.log"
dbfilename "dump-7000.rdb"
cluster-enabled yes # 开启集群模式
cluster-config-file nodes-7000.conf # 集群内部的配置文件
cluster-require-full-coverage no # Redis Cluster需要16384个slot都正常的时候才能对外提供服务。换句话说只要任何一个slot异常那么整个cluster不对外提供服务,因此生产环境一般为no
redis支持多实例的功能,我们在单机演示集群搭建,需要6个实例,3个是主节点,3个是从节点,数量为6个节点才能保证高可用的集群
每个节点仅仅是端口运行的不同
[root@web01 /opt/redis/config]# ls
redis-7000.conf redis-7002.conf redis-7004.conf
redis-7001.conf redis-7003.conf redis-7005.conf
# 确保每个配置文件中的端口修改!!
2.运行redis实例
创建6个节点的redis实例
1855 2021-07-24 15:46:01 redis-server redis-7000.conf
1856 2021-07-24 15:46:13 redis-server redis-7001.conf
1857 2021-07-24 15:46:16 redis-server redis-7002.conf
1858 2021-07-24 15:46:18 redis-server redis-7003.conf
1859 2021-07-24 15:46:20 redis-server redis-7004.conf
1860 2021-07-24 15:46:23 redis-server redis-7005.conf
检查日志文件
cat 7000.log
检查redis服务的端口、进程
netstat -tunlp|grep redis
ps -ef|grep redis
如果此时的集群还不可用的话,可以通过登录Redis查看
redis-cli -p 7000
set hello world
(error)CLUSTERDOWN The cluster is down
3.创建开启redis-cluster
准备ruby环境
1、下载、编译、安装Ruby
2、安装rubygem redis
3、安装redis-trib.rb命令
第一步,安装ruby ( 这些命令可以放入一个sh脚本文件里 )
#下载ruby
wget https://cache.ruby-lang.org/pub/ruby/2.3/ruby-2.3.1.tar.gz
# 安装ruby
tar -xvf ruby-2.3.1.tar.gz
./configure --prefix=/opt/ruby/
make && make install
# 准备一个ruby命令#准备一个gem软件包管理命令
# 拷贝ruby命令到path下/usr/local/ruby
cp /opt/ruby/bin/ruby /usr/local/cp bin/gem /usr/local/bin
安装ruby gem 包管理工具
wget http://rubygems.org/downloads/redis-3.3.0.gem
gem install -l redis-3.3.0.gem
# 查看gem有哪些包
gem list -- check redis gem
安装redis-trib.rb命令
[root@web01 /opt/redis/src]# cp /opt/redis/src/redis-trib.rb /usr/local/bin/
一键开启redis-cluster集群
# 每个主节点,有一个从节点,代表--replicas 1
redis-trib.rb create --replicas 1 127.0.0.1:7000 127.0.0.1:7001 127.0.0.1:7002 127.0.0.1:7003 127.0.0.1:7004 127.0.0.1:7005
# 集群自动分配主从关系:7000、7001、7002为 7003、7004、7005 主动关系
查看集群状态
redis-cli -p 7000 cluster info
redis-cli -p 7000 cluster nodes # 等同于查看nodes-7000.conf 文件节点信息
# 集群主节点状态
redis-cli -p 7000 cluster nodes | grep master
# 集群从节点状态
redis-cli -p 7000 cluster nodes | grep slave
安装完毕后,检查集群状态
[root@web01 /opt/redis/src]# redis-cli -p 7000 cluster info
cluster_state:ok
cluster_slots_assigned:16384
cluster_slots_ok:16384
cluster_slots_pfail:0
cluster_slots_fail:0
cluster_known_nodes:6
cluster_size:3
cluster_current_epoch:6
cluster_my_epoch:1
cluster_stats_messages_ping_sent:10468
cluster_stats_messages_pong_sent:10558
cluster_stats_messages_sent:21026
cluster_stats_messages_ping_received:10553
cluster_stats_messages_pong_received:10468
cluster_stats_messages_meet_received:5
cluster_stats_messages_received:21026
测试写入集群数据,登录集群必须使用redis-cli -c -p 7000必须加上-c参数
127.0.0.1:7000> set name chao
-> Redirected to slot [5798] located at 127.0.0.1:7001
OK
127.0.0.1:7001> exit
[root@web01 /opt/redis/src]# redis-cli -c -p 7000
127.0.0.1:7000> ping
PONG
127.0.0.1:7000> keys *
(empty list or set)
127.0.0.1:7000> get name
-> Redirected to slot [5798] located at 127.0.0.1:7001
"chao"
集群ok
工作原理:
Redis客户端任意访问一个Redis实例,如果数据不在该实例中的话,可以通过重定向引导客户端,访问所需要的Redis实例。
Redis-Cli的各种参数用法
-r
重复执行次数。
-i
间隔时间,单位是秒。
-x
代表从标准输入读取数据作为,redis-cli的最后一个参数。
-c
连接RedisCluster的参数,可以防止moved和ask异常。
-a
Redis的密码。
–scan和–pattern
用于扫描指定模式的键,相当于使用scan命令。
–slave
把当前客户端模拟成当前Redis节点的从节点,可以用来获取当前Redis节点的更新操作。
–rdb
会请求Redis实例生成并发送RDB持久化文件,保存在本地,可用它做持久化文件的定期备份。
–pipe
用于将命令封装成Redis通信协议定义的数据格式,批量发送给Redis执行
–bigkeys
使用scan命令对Redis的键进行采样,从中找到内存占用比较大的键值,这些键可能是系统的瓶颈。
–eval
用于执行指定Lua脚本。
–latency
可以测试从客户端到目标Redis的网络延迟。
–latency-history
可以获取历史上每15秒的统计结果。可以通过-i参数,控制间隔。
–latency-dist
会使用统计图表的形式,从控制台输出延迟统计信息。
–stat
可以实时获取Redis的重要统计信息,且可以实时看到一些增量的数据。
–raw和–no-raw
–raw
可以返回格式化的结果,如中文可以正确显示,而不是二进制形式。
–no-raw
与raw相反,返回二进制形式。
网友评论