一、前言
redis在我們企業級開發中是很常見的,但是單個redis不能保證我們的穩定使用,是以我們要建立一個叢集。
redis有兩種高可用的方案:
- High availability with Redis Sentinel(哨兵)
- Scaling with Redis Cluster(分片叢集)
第一個就是我們本次的要搭建的,就是高可用的哨兵,主redis挂掉,哨兵會進行投票進行故障轉移!
第二個就是分片叢集,哨兵的一個缺點就是隻能存在一個master節點,寫的效率太低。分片叢集就是解決哨兵的問題,可以水準擴充,提高redis的性能!
哨兵最低配是三哨兵,以奇數遞增。
分片叢集最低配是三主三從。
小編之前寫過一篇在一台機器上搭建的文章,大家有興趣可以先去體驗一下,實際生産上不會讓你一台機器上;也沒有任何意義,伺服器挂了,再多的叢集也全部挂掉了!!
docker compose搭建redis7.0.4高可用一主二從三哨兵叢集并整合SpringBoot【圖文完整版】
二、準備
首先我們要準備:
三台伺服器(沒有的條件的搭三個虛拟機),巧了小編就是虛拟機哈!
三台機器的ip和名稱在表格裡整理一下!
ip | redis節點名稱 | sentinel節點名稱 |
192.168.239.131 | redis-master | redis-sentinel-1 |
192.168.239.130 | redis-slave-1 | redis-sentinel-2 |
192.168.239.128 | redis-slave-2 | redis-sentinel-3 |
三、Sentinel概念
Redis Sentinel 在不使用Redis Cluster時為 Redis 提供高可用性。
Sentinel功能的完整清單:
- 監控:Sentinel 不斷檢查您的主執行個體和副本執行個體是否按預期工作。
- 通知:Sentinel 可以通過 API 通知系統管理者或其他計算機程式,其中一個受監控的 Redis 執行個體出現問題。
- 自動故障轉移:如果 master 沒有按預期工作,Sentinel 可以啟動一個故障轉移過程,其中一個副本被提升為 master,其他額外的副本被重新配置為使用新的 master,并且使用 Redis 伺服器的應用程式被告知要使用的新位址連接配接時。
- 自動更新配置:Sentinel 充當用戶端服務發現的權威來源:用戶端連接配接到 Sentinels 以詢問負責給定服務的目前 Redis master 的位址。如果發生故障轉移,Sentinels 将報告新位址。
官方哨兵搭建條件:
- 您至少需要三個 Sentinel 執行個體才能進行可靠的部署。
- 三個 Sentinel 執行個體應該放置在被認為以獨立方式發生故障的計算機或虛拟機中。是以,例如在不同可用區上執行的不同實體伺服器或虛拟機。
詳細介紹和使用:請見官網 --->官網詳細文檔:https://redis.io/docs/management/sentinel/
四、一主二從搭建
話不多說,咱們直接開始搭建哈!
1. 建立挂載目錄
三台機器上建立目錄:
首先我們開啟三個xshell視窗,然後同時操作三個視窗建立
在這裡插入圖檔描述
然後再左下角選擇發送到全部視窗!
cd /
mkdir mydata
cd /mydata
mkdir redis
cd redis
mkdir data
mkdir conf
cd conf
2. 在`192.168.239.131`機器上編輯檔案
vim redis.conf
輸入以下内容:
# 任何都可以連接配接redis
bind 0.0.0.0
# 配置master密碼
requirepass 123456
# 當機後成為從要連接配接master的密碼
masterauth 123456
# 開啟持久化
appendonly yes
在這裡插入圖檔描述
3. 在`192.168.239.130`機器上編輯檔案
vim redis.conf
輸入以下内容:
# 配置master的ip和端口号
replicaof 192.168.239.131 6379
# 任何都可以連接配接redis
bind 0.0.0.0
# 成為master後的密碼
requirepass 123456
# 連接配接master密碼
masterauth 123456
# 開啟持久化
appendonly yes
在這裡插入圖檔描述
4. 在`192.168.239.128`機器上編輯檔案
vim redis.conf
輸入以下内容:
# 配置master的ip和端口号
replicaof 192.168.239.131 6379
# 任何都可以連接配接redis
bind 0.0.0.0
# 成為master後的密碼
requirepass 123456
# 連接配接master密碼
masterauth 123456
# 開啟持久化
appendonly yes
5. `192.168.239.131`啟動redis
docker run -p 6379:6379 --name redis-master \
-v /mydata/redis/data:/usr/local/etc/redis/data \
-v /mydata/redis/conf/redis.conf:/usr/local/etc/redis/redis.conf \
-d redis redis-server /usr/local/etc/redis/redis.conf
檢視啟動日志:
docker logs -f redis-master
在這裡插入圖檔描述
6. `192.168.239.130`啟動redis
docker run -p 6379:6379 --name redis-slave-1 \
-v /mydata/redis/data:/usr/local/etc/redis/data \
-v /mydata/redis/conf/redis.conf:/usr/local/etc/redis/redis.conf \
-d redis redis-server /usr/local/etc/redis/redis.conf
可以看到已經連接配接到master節點了!
在這裡插入圖檔描述
7. `192.168.239.128`啟動redis
docker run -p 6379:6379 --name redis-slave-2 \
-v /mydata/redis/data:/usr/local/etc/redis/data \
-v /mydata/redis/conf/redis.conf:/usr/local/etc/redis/redis.conf \
-d redis redis-server /usr/local/etc/redis/redis.conf
在這裡插入圖檔描述
8. 進入`192.168.239.130`容器檢視
我們檢視master日志,可以看到兩個從節點已經加入進來了!
在這裡插入圖檔描述
我們進入容器進行在次檢視:
docker exec -it redis-master /bin/bash
連接配接redis:
redis-cli
登入redis
auth 123456
在這裡插入圖檔描述
檢視從節點:
info
也是可以看到有兩個從節點!
在這裡插入圖檔描述
五、搭建三哨兵sentinel
1. 建立挂載目錄
還是三個虛拟機一起建立
mkdir sentinel
cd sentinel
vim sentinel.conf
輸入下面内容:
port 26379
sentinel monitor redis-master 192.168.239.131 6379 2
sentinel auth-pass redis-master 123456
sentinel down-after-milliseconds redis-master 6000
sentinel parallel-syncs redis-master 1
sentinel failover-timeout redis-master 6000
第二行:Redis 監控一個名為redis-master的redis叢集,我們可以随意寫;後面就是ip,我們主控端的ip即可,端口為主redis的端口;2為哨兵投票的票數,當主redis當機,三個哨兵必須兩個哨兵都投票的redis才會變為主!!
第三行:配置master的密碼
第四行:Sentinel判斷執行個體進入主觀下線所需的時間,毫秒機關。
第五行:限制在一次故障轉移之後,每次向新的主節點同時發起複制操作節點個數,越大效率越慢。
第六行:在指定的時間内未能完成failover故障轉移,則任務故障轉移失敗。
在這裡插入圖檔描述
2. 運作`192.168.239.131`哨兵
docker run -p 26379:26379 --name redis-sentinel-1 \
-v /mydata/redis/sentinel/sentinel.conf:/usr/local/etc/redis/sentinel.conf \
-d redis redis-sentinel /usr/local/etc/redis/sentinel.conf
3. 運作`192.168.239.130`哨兵
docker run -p 26379:26379 --name redis-sentinel-2 \
-v /mydata/redis/sentinel/sentinel.conf:/usr/local/etc/redis/sentinel.conf \
-d redis redis-sentinel /usr/local/etc/redis/sentinel.conf
3. 運作`192.168.239.128`哨兵
docker run -p 26379:26379 --name redis-sentinel-3 \
-v /mydata/redis/sentinel/sentinel.conf:/usr/local/etc/redis/sentinel.conf \
-d redis redis-sentinel /usr/local/etc/redis/sentinel.conf
4. 待解決問題
這裡建立三個哨兵,檢視卻是有四個,不知道什麼問題,換了虛拟機還是不行!有大佬懂的可以分享一下哈!!
在這裡插入圖檔描述
六、測試主從和故障轉移
1. 測試主從複制
master節點建立一個鍵值對:
set a b
在這裡插入圖檔描述
從檢視key是否存在:
get a
在這裡插入圖檔描述
主從沒有問題哈!
2. 測試故障轉移
我們把master停掉,檢視一個哨兵的日志:
docker stop redis-master
docker logs -f redis-sentinel-1
我們看到192.168.239.130成為master!
在這裡插入圖檔描述
重新啟動原來的master:
docker restart redis-master
在這裡插入圖檔描述
故障轉移成功!!
七、整合springboot
1. 導入依賴
小編的springboot版本為:2.7.4
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
2. yml配置
password密碼和sentinel同級,不然找不到密碼,驗證失敗
server:
port: 8087
spring:
redis:
# 密碼和sentinel同級,不然找不到密碼,驗證失敗
password: 123456
sentinel:
# sentinel.conf裡的叢集名稱
master: my-master
# 我們隻需要連哨兵即可,哨兵内部會幫我們找到redis
nodes:
- 192.168.239.131:26379
- 192.168.239.130:26379
- 192.168.239.128:26379
3. json序列化配置
/**
* @author wangzhenjun
* @date 2022/11/24 10:37
*/
@Configuration
public class RedisConfig {
@Bean
@SuppressWarnings(value = { "unchecked", "rawtypes" })
public RedisTemplate<Object, Object> redisTemplate(RedisConnectionFactory connectionFactory)
{
RedisTemplate<Object, Object> template = new RedisTemplate<>();
template.setConnectionFactory(connectionFactory);
Jackson2JsonRedisSerializer serializer = new Jackson2JsonRedisSerializer(Object.class);
// 使用StringRedisSerializer來序列化和反序列化redis的key值
template.setKeySerializer(new StringRedisSerializer());
template.setValueSerializer(serializer);
// Hash的key也采用StringRedisSerializer的序列化方式
template.setHashKeySerializer(new StringRedisSerializer());
template.setHashValueSerializer(serializer);
template.afterPropertiesSet();
return template;
}
}
4. 建立controller測試
/**
* @author wangzhenjun
* @date 2022/11/24 10:37
*/
@RestController
public class TestController {
@Autowired
private RedisTemplate redisTemplate;
@GetMapping("/redis")
public void saveRedis(){
redisTemplate.opsForValue().set("test","看到我就成功了");
}
}
5. 測試
http://localhost:8087/test/redis
在這裡插入圖檔描述
6. 檢視redis
在這裡插入圖檔描述
八、總結
經過一天的搭建,終于完成了,雖然不是完美的,但是大體功能是沒有問題的!但是不影響故障轉移和主從複制!
唯一的遺憾:三個哨兵,檢視就是四個!從第三個哨兵加入後變為4個!
如果對你有幫助,還請不要吝啬您的發财小手,你的一鍵三連是我寫作的動力,謝謝大家哈!!
可以看下小編的微信公衆号,和網站文章首發看,歡迎關注,一起交流哈!!