在inux中搭建并使用Redis集群。
Redis 集群是一个提供在多个Redis间节点间共享数据的程序集。
Redis集群并不支持处理多个keys的命令,因为这需要在不同的节点间移动数据,从而达不到像Redis那样的性能,在高负载的情况下可能会导致不可预料的错误.
Redis 集群通过分区来提供一定程度的可用性,在实际环境中当某个节点宕机或者不可达的情况下继续处理命令. Redis 集群的优势:
我这里将搭建一个3主3从的redis集群,共使用2台机器。
下面开始redis集群的安装搭建
在根目录下创建文件夹
mkdir app
下载redis(两台机器中都需要下载)
wget http://download.redis.io/releases/redis-5.0.3.tar.gz
解压redis安装文件
tar -zxvf redis-5.0.3.tar.gz
进入redis目录
cd redis-5.0.3
编译
make
出现以下错误则需要安装gcc
make[3]: gcc: Command not found
make[3]: *** [net.o] Error 127
make[3]: Leaving directory /app/redis-5.0.3/deps/hiredis' make[2]: *** [hiredis] Error 2 make[2]: Leaving directory
/app/redis-5.0.3/deps’
make[1]: [persist-settings] Error 2 (ignored)
CC adlist.o
/bin/sh: cc: command not found
make[1]: *** [adlist.o] Error 127
make[1]: Leaving directory `/app/redis-5.0.3/src’
make: *** [all] Error 2
yum -y install gcc
再次make碰到以下错误
zmalloc.h:50:31: fatal error: jemalloc/jemalloc.h: No such file or directory
#include
^
compilation terminated.
make[1]: *** [adlist.o] Error 1
make[1]: Leaving directory `/app/redis-5.0.3/src’
make: *** [all] Error 2
解决:使用 make MALLOC=libc
命令进行编译
创建redis集群目录:
mkdir /usr/local/redis-cluster
进入 redis-cluster 目录
cd /usr/local/redis-cluster
在 10.80.9.65 机器中创建 7001 7002 7003 目录
mkdir 7001 7002 7003
在 10.80.9.69 机器中创建 7004 7005 7006 目录
mkdir 7004 7005 7006
将redis.conf文件复制到 7001 - 7006目录中
redis.conf 修改成以下内容:
#端口7001,7002,7003
port 7001
#本机ip
bind 10.80.9.65
#redis后台运行
daemonize yes
#pidfile文件对应7001,7002,7003
pidfile /usr/local/redis-cluster/redis-7001.pid
#开启集群
cluster-enabled yes
#集群的配置 配置文件首次启动自动生成 7000,7001,7002
cluster-config-file /usr/local/redis-cluster/nodes-7001.conf
#请求超时 默认15秒,可自行设置
cluster-node-timeout 15000
#aof日志开启 有需要就开启,它会每次写操作都记录一条日志
appendonly no
7002 - 7006 也是同样的内容,只需修改对应的ip端口等。
#引入redis公用的配置文件
include /usr/local/redis-cluster/etc/redis-common.conf
port 7001
#redis持久化文件存放路径
dir /data/redis-cluster/7001/
pidfile /usr/local/redis-cluster/7001/redis-7001.pid
logfile /usr/local/redis-cluster/7001/redis-7001.log
cluster-config-file /usr/local/redis-cluster/7001/nodes-7001.conf
redis-common.conf 中的配置,基本都是redis的默认值
bind 10.80.9.65
protected-mode yes
#密码
#masterauth "test123"
#requirepass "test123"
tcp-backlog 511
timeout 0
tcp-keepalive 300
daemonize yes
supervised no
loglevel notice
databases 16
always-show-logo yes
save 900 1
save 300 10
save 60 10000
stop-writes-on-bgsave-error yes
rdbcompression yes
rdbchecksum yes
dbfilename dump.rdb
replica-serve-stale-data yes
replica-read-only yes
repl-diskless-sync no
repl-diskless-sync-delay 5
repl-disable-tcp-nodelay no
replica-priority 100
lazyfree-lazy-eviction no
lazyfree-lazy-expire no
lazyfree-lazy-server-del no
replica-lazy-flush no
appendonly no
appendfilename "appendonly.aof"
appendfsync everysec
no-appendfsync-on-rewrite no
auto-aof-rewrite-percentage 100
auto-aof-rewrite-min-size 64mb
aof-load-truncated yes
aof-use-rdb-preamble yes
lua-time-limit 5000
cluster-enabled yes
cluster-node-timeout 15000
slowlog-log-slower-than 10000
slowlog-max-len 128
latency-monitor-threshold 0
notify-keyspace-events ""
hash-max-ziplist-entries 512
hash-max-ziplist-value 64
list-max-ziplist-size -2
list-compress-depth 0
set-max-intset-entries 512
zset-max-ziplist-entries 128
zset-max-ziplist-value 64
hll-sparse-max-bytes 3000
stream-node-max-bytes 4096
stream-node-max-entries 100
activerehashing yes
client-output-buffer-limit normal 0 0 0
client-output-buffer-limit replica 256mb 64mb 60
client-output-buffer-limit pubsub 32mb 8mb 60
hz 10
dynamic-hz yes
aof-rewrite-incremental-fsync yes
rdb-save-incremental-fsync yes
机器1:
/app/redis-5.0.3/src/redis-server /usr/local/redis-cluster/7001/redis-7001.conf
/app/redis-5.0.3/src/redis-server /usr/local/redis-cluster/7002/redis-7002.conf
/app/redis-5.0.3/src/redis-server /usr/local/redis-cluster/7003/redis-7003.conf
查看是否启动
ps -ef|grep redis
/app/redis-5.0.3/src/redis-server /usr/local/redis-cluster/7004/redis-7004.conf
/app/redis-5.0.3/src/redis-server /usr/local/redis-cluster/7005/redis-7005.conf
/app/redis-5.0.3/src/redis-server /usr/local/redis-cluster/7006/redis-7006.conf
查看是否启动
ps -ef|grep redis
/app/redis-5.0.3/src/redis-cli --cluster create 10.80.9.65:7001 10.80.9.65:7002 10.80.9.65:7003 10.80.9.69:7004 10.80.9.69:7005 10.80.9.69:7006 --cluster-replicas 1
出现以下错误:
No route to host
清除防火墙策略,正式环境请谨慎操作以下命令
iptables -F
Adding replica 10.80.9.69:7005 to 10.80.9.65:7001
Adding replica 10.80.9.65:7003 to 10.80.9.69:7004
Adding replica 10.80.9.69:7006 to 10.80.9.65:7002
集群的启停
redis 5 提供了关闭集群的工具,在如下目录:
/app/redis-5.0.3/utils/create-cluster
打开此文件修改端口为我们自己的,如下所示:
端口PROT设置为7000,NODES为6,工具会自动累加1 生成 7001-7006 六个节点 用于操作。
修改后集群关闭命令如下:
/app/redis-5.0.3/utils/create-cluster/create-cluster stop
启动命令
/app/redis-5.0.3/utils/create-cluster/create-cluster start
也可以自己编写脚本进行启动:
机器1:
#!/bin/sh
if [ `ps -ef|grep redis-server|grep -v grep|wc -l` -gt 0 ] ; then
ps -ef|grep redis-server|grep -v grep|awk '{print $2}'|xargs kill -9
fi
/app/redis-5.0.3/src/redis-server /usr/local/redis-cluster/7001/redis-7001.conf
/app/redis-5.0.3/src/redis-server /usr/local/redis-cluster/7002/redis-7002.conf
/app/redis-5.0.3/src/redis-server /usr/local/redis-cluster/7003/redis-7003.conf
机器2:
#!/bin/sh
if [ `ps -ef|grep redis-server|grep -v grep|wc -l` -gt 0 ] ; then
ps -ef|grep redis-server|grep -v grep|awk '{print $2}'|xargs kill -9
fi
/app/redis-5.0.3/src/redis-server /usr/local/redis-cluster/7004/redis-7004.conf
/app/redis-5.0.3/src/redis-server /usr/local/redis-cluster/7005/redis-7005.conf
/app/redis-5.0.3/src/redis-server /usr/local/redis-cluster/7006/redis-7006.conf
将上面内容复制到脚本中,对应机器
vim .startRedis.sh
cd $home
vim .bash_profile
alias startRedis='sh /usr/local/redis-cluster/.startRedis.sh'
source .bash_profile
直接输入 startRedis命令就可以启动本台机器上的redis节点
在第一台机器上连接集群的7002端口的节点,在另外一台连接7006节点,连接方式为 /app/redis-5.0.3/src/redis-cli -h 10.80.9.65 -c -p 7002
,加参数 -C 可连接到集群,因为上面 redis.conf 将 bind 改为了ip地址,所以 -h 参数不可以省略。
set hello word
这个键插入到了7001节点, 7001节点的从节点是10.80.9.69 7005节点,数据也会自动插入到从节点,我们可以验证一下。
简单说一下原理
redis cluster在设计的时候,就考虑到了去中心化,去中间件,也就是说,集群中的每个节点都是平等的关系,都是对等的,每个节点都保存各自的数据和整个集群的状态。每个节点都和其他所有节点连接,而且这些连接保持活跃,这样就保证了我们只需要连接集群中的任意一个节点,就可以获取到其他节点的数据。
Redis 集群没有并使用传统的一致性哈希来分配数据,而是采用另外一种叫做哈希槽 (hash slot)的方式来分配的。redis cluster 默认分配了 16384 个slot,当我们set一个key 时,会用CRC16算法来取模得到所属的slot,然后将这个key 分到哈希槽区间的节点上,具体算法就是:CRC16(key) % 16384。所以我们在测试的时候看到set 和 get 的时候,直接跳转到了7001端口的节点。
Redis 集群会把数据存在一个 master 节点,然后在这个 master 和其对应的salve 之间进行数据同步。当读取数据时,也根据一致性哈希算法到对应的 master 节点获取数据。只有当一个master 挂掉之后,才会启动一个对应的 salve 节点,充当 master 。
需要注意的是:必须要3个或以上的主节点,否则在创建集群时会失败,并且当存活的主节点数小于总节点数的一半时,整个集群就无法提供服务了。