mirror of
https://github.com/zhwei820/learn.lianglianglee.com.git
synced 2025-09-25 04:36:41 +08:00
670 lines
29 KiB
HTML
670 lines
29 KiB
HTML
<!DOCTYPE html>
|
||
<!-- saved from url=(0046)https://kaiiiz.github.io/hexo-theme-book-demo/ -->
|
||
<html xmlns="http://www.w3.org/1999/xhtml">
|
||
<head>
|
||
<head>
|
||
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
|
||
<meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1.0, user-scalable=no">
|
||
<link rel="icon" href="/static/favicon.png">
|
||
<title>38 实战:Redis 哨兵模式(下).md.html</title>
|
||
<!-- Spectre.css framework -->
|
||
<link rel="stylesheet" href="/static/index.css">
|
||
<!-- theme css & js -->
|
||
<meta name="generator" content="Hexo 4.2.0">
|
||
</head>
|
||
<body>
|
||
<div class="book-container">
|
||
<div class="book-sidebar">
|
||
<div class="book-brand">
|
||
<a href="/">
|
||
<img src="/static/favicon.png">
|
||
<span>技术文章摘抄</span>
|
||
</a>
|
||
</div>
|
||
<div class="book-menu uncollapsible">
|
||
<ul class="uncollapsible">
|
||
<li><a href="/" class="current-tab">首页</a></li>
|
||
</ul>
|
||
<ul class="uncollapsible">
|
||
<li><a href="../">上一级</a></li>
|
||
</ul>
|
||
<ul class="uncollapsible">
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/01 Redis 是如何执行的.md.html">01 Redis 是如何执行的</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/02 Redis 快速搭建与使用.md.html">02 Redis 快速搭建与使用</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/03 Redis 持久化——RDB.md.html">03 Redis 持久化——RDB</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/04 Redis 持久化——AOF.md.html">04 Redis 持久化——AOF</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/05 Redis 持久化——混合持久化.md.html">05 Redis 持久化——混合持久化</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/06 字符串使用与内部实现原理.md.html">06 字符串使用与内部实现原理</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/07 附录:更多字符串操作命令.md.html">07 附录:更多字符串操作命令</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/08 字典使用与内部实现原理.md.html">08 字典使用与内部实现原理</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/09 附录:更多字典操作命令.md.html">09 附录:更多字典操作命令</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/10 列表使用与内部实现原理.md.html">10 列表使用与内部实现原理</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/11 附录:更多列表操作命令.md.html">11 附录:更多列表操作命令</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/12 集合使用与内部实现原理.md.html">12 集合使用与内部实现原理</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/13 附录:更多集合操作命令.md.html">13 附录:更多集合操作命令</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/14 有序集合使用与内部实现原理.md.html">14 有序集合使用与内部实现原理</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/15 附录:更多有序集合操作命令.md.html">15 附录:更多有序集合操作命令</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/16 Redis 事务深入解析.md.html">16 Redis 事务深入解析</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/17 Redis 键值过期操作.md.html">17 Redis 键值过期操作</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/18 Redis 过期策略与源码分析.md.html">18 Redis 过期策略与源码分析</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/19 Redis 管道技术——Pipeline.md.html">19 Redis 管道技术——Pipeline</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/20 查询附近的人——GEO.md.html">20 查询附近的人——GEO</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/21 游标迭代器(过滤器)——Scan.md.html">21 游标迭代器(过滤器)——Scan</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/22 优秀的基数统计算法——HyperLogLog.md.html">22 优秀的基数统计算法——HyperLogLog</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/23 内存淘汰机制与算法.md.html">23 内存淘汰机制与算法</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/24 消息队列——发布订阅模式.md.html">24 消息队列——发布订阅模式</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/25 消息队列的其他实现方式.md.html">25 消息队列的其他实现方式</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/26 消息队列终极解决方案——Stream(上).md.html">26 消息队列终极解决方案——Stream(上)</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/27 消息队列终极解决方案——Stream(下).md.html">27 消息队列终极解决方案——Stream(下)</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/28 实战:分布式锁详解与代码.md.html">28 实战:分布式锁详解与代码</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/29 实战:布隆过滤器安装与使用及原理分析.md.html">29 实战:布隆过滤器安装与使用及原理分析</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/30 完整案例:实现延迟队列的两种方法.md.html">30 完整案例:实现延迟队列的两种方法</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/31 实战:定时任务案例.md.html">31 实战:定时任务案例</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/32 实战:RediSearch 高性能的全文搜索引擎.md.html">32 实战:RediSearch 高性能的全文搜索引擎</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/33 实战:Redis 性能测试.md.html">33 实战:Redis 性能测试</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/34 实战:Redis 慢查询.md.html">34 实战:Redis 慢查询</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/35 实战:Redis 性能优化方案.md.html">35 实战:Redis 性能优化方案</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/36 实战:Redis 主从同步.md.html">36 实战:Redis 主从同步</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/37 实战:Redis哨兵模式(上).md.html">37 实战:Redis哨兵模式(上)</a>
|
||
</li>
|
||
<li>
|
||
<a class="current-tab" href="/专栏/Redis 核心原理与实战/38 实战:Redis 哨兵模式(下).md.html">38 实战:Redis 哨兵模式(下)</a>
|
||
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/39 实战:Redis 集群模式(上).md.html">39 实战:Redis 集群模式(上)</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/40 实战:Redis 集群模式(下).md.html">40 实战:Redis 集群模式(下)</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/41 案例:Redis 问题汇总和相关解决方案.md.html">41 案例:Redis 问题汇总和相关解决方案</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/42 技能学习指南.md.html">42 技能学习指南</a>
|
||
</li>
|
||
<li>
|
||
<a href="/专栏/Redis 核心原理与实战/43 加餐:Redis 的可视化管理工具.md.html">43 加餐:Redis 的可视化管理工具</a>
|
||
</li>
|
||
</ul>
|
||
</div>
|
||
</div>
|
||
<div class="sidebar-toggle" onclick="sidebar_toggle()" onmouseover="add_inner()" onmouseleave="remove_inner()">
|
||
<div class="sidebar-toggle-inner"></div>
|
||
</div>
|
||
<script>
|
||
function add_inner() {
|
||
let inner = document.querySelector('.sidebar-toggle-inner')
|
||
inner.classList.add('show')
|
||
}
|
||
function remove_inner() {
|
||
let inner = document.querySelector('.sidebar-toggle-inner')
|
||
inner.classList.remove('show')
|
||
}
|
||
function sidebar_toggle() {
|
||
let sidebar_toggle = document.querySelector('.sidebar-toggle')
|
||
let sidebar = document.querySelector('.book-sidebar')
|
||
let content = document.querySelector('.off-canvas-content')
|
||
if (sidebar_toggle.classList.contains('extend')) { // show
|
||
sidebar_toggle.classList.remove('extend')
|
||
sidebar.classList.remove('hide')
|
||
content.classList.remove('extend')
|
||
} else { // hide
|
||
sidebar_toggle.classList.add('extend')
|
||
sidebar.classList.add('hide')
|
||
content.classList.add('extend')
|
||
}
|
||
}
|
||
function open_sidebar() {
|
||
let sidebar = document.querySelector('.book-sidebar')
|
||
let overlay = document.querySelector('.off-canvas-overlay')
|
||
sidebar.classList.add('show')
|
||
overlay.classList.add('show')
|
||
}
|
||
function hide_canvas() {
|
||
let sidebar = document.querySelector('.book-sidebar')
|
||
let overlay = document.querySelector('.off-canvas-overlay')
|
||
sidebar.classList.remove('show')
|
||
overlay.classList.remove('show')
|
||
}
|
||
</script>
|
||
<div class="off-canvas-content">
|
||
<div class="columns">
|
||
<div class="column col-12 col-lg-12">
|
||
<div class="book-navbar">
|
||
<!-- For Responsive Layout -->
|
||
<header class="navbar">
|
||
<section class="navbar-section">
|
||
<a onclick="open_sidebar()">
|
||
<i class="icon icon-menu"></i>
|
||
</a>
|
||
</section>
|
||
</header>
|
||
</div>
|
||
<div class="book-content" style="max-width: 960px; margin: 0 auto;
|
||
overflow-x: auto;
|
||
overflow-y: hidden;">
|
||
<div class="book-post">
|
||
<p id="tip" align="center"></p>
|
||
<div><h1>38 实战:Redis 哨兵模式(下)</h1>
|
||
<p>上一篇我们介绍了 Redis Sentinel 的搭建和运行原理,本文我们重点来看下 Sentinel 的命令操作和代码实战。</p>
|
||
<h3>Sentinel 命令操作</h3>
|
||
<p>要使用 Sentinel 实现要连接到 Sentinel 服务器,和连接 Redis 服务相同,我们可以使用 redis-cli 来连接 Sentinel,如下命令所示:</p>
|
||
<pre><code class="language-shell">[@iZ2ze0nc5n41zomzyqtksmZ:~]$ redis-cli -h 127.0.0.1 -p 26379 -a pwd654321
|
||
127.0.0.1:26379> ping
|
||
PONG
|
||
</code></pre>
|
||
<p>其中:</p>
|
||
<ul>
|
||
<li>-h 后面输入的是 Sentinel 的 IP;</li>
|
||
<li>-p 后面输入的是 Sentinel 的端口,默认是 26379;</li>
|
||
<li>-a 后面输入的是密码。</li>
|
||
</ul>
|
||
<p>Sentinel 的端口号可以在 sentinel.conf 里面配置,通过 port 选项设置。</p>
|
||
<p>注意:<strong>Sentinel 可以监视多台主节点,而不是只能监视一台服务器</strong>。想要监视多台主节点只需要在配置文件中设置多个 <code>sentinel monitor master-name ip port quorum</code> 即可,我们通过 master-name 来区分不同的主节点。</p>
|
||
<h4><strong>查询所有被监控的主服务器信息</strong></h4>
|
||
<pre><code class="language-shell">127.0.0.1:26379> sentinel masters
|
||
1) 1) "name"
|
||
2) "mymaster"
|
||
3) "ip"
|
||
4) "127.0.0.1"
|
||
5) "port"
|
||
6) "6377"
|
||
7) "runid"
|
||
8) "eb3552c6fc8974f91466c4ada90fe23ef30fd89c"
|
||
9) "flags"
|
||
10) "master"
|
||
11) "link-pending-commands"
|
||
12) "0"
|
||
13) "link-refcount"
|
||
14) "1"
|
||
15) "last-ping-sent"
|
||
16) "0"
|
||
17) "last-ok-ping-reply"
|
||
18) "400"
|
||
19) "last-ping-reply"
|
||
20) "400"
|
||
21) "down-after-milliseconds"
|
||
22) "30000"
|
||
23) "info-refresh"
|
||
24) "5731"
|
||
25) "role-reported"
|
||
26) "master"
|
||
27) "role-reported-time"
|
||
28) "75963321"
|
||
29) "config-epoch"
|
||
30) "7"
|
||
31) "num-slaves"
|
||
32) "2"
|
||
33) "num-other-sentinels"
|
||
34) "1"
|
||
35) "quorum"
|
||
36) "2"
|
||
37) "failover-timeout"
|
||
38) "180000"
|
||
39) "parallel-syncs"
|
||
40) "1"
|
||
</code></pre>
|
||
<p>相关语法:</p>
|
||
<pre><code>sentinel masters
|
||
</code></pre>
|
||
<p>因为我们配置的 Sentinel 只监视了一台主服务器,所以只有一台服务器的信息。</p>
|
||
<h4><strong>查询某个主节点的信息</strong></h4>
|
||
<pre><code class="language-shell">127.0.0.1:26379> sentinel master mymaster
|
||
1) "name"
|
||
2) "mymaster"
|
||
3) "ip"
|
||
4) "127.0.0.1"
|
||
5) "port"
|
||
6) "6377"
|
||
7) "runid"
|
||
8) "eb3552c6fc8974f91466c4ada90fe23ef30fd89c"
|
||
9) "flags"
|
||
10) "master"
|
||
11) "link-pending-commands"
|
||
12) "0"
|
||
13) "link-refcount"
|
||
14) "1"
|
||
15) "last-ping-sent"
|
||
16) "0"
|
||
17) "last-ok-ping-reply"
|
||
18) "250"
|
||
19) "last-ping-reply"
|
||
20) "250"
|
||
21) "down-after-milliseconds"
|
||
22) "30000"
|
||
23) "info-refresh"
|
||
24) "8191"
|
||
25) "role-reported"
|
||
26) "master"
|
||
27) "role-reported-time"
|
||
28) "76096303"
|
||
29) "config-epoch"
|
||
30) "7"
|
||
31) "num-slaves"
|
||
32) "2"
|
||
33) "num-other-sentinels"
|
||
34) "1"
|
||
35) "quorum"
|
||
36) "2"
|
||
37) "failover-timeout"
|
||
38) "180000"
|
||
39) "parallel-syncs"
|
||
40) "1"
|
||
</code></pre>
|
||
<p>相关语法:</p>
|
||
<pre><code>sentinel master master-name
|
||
</code></pre>
|
||
<h4><strong>查看某个主节点的 IP 和端口</strong></h4>
|
||
<pre><code class="language-shell">127.0.0.1:26379> sentinel get-master-addr-by-name mymaster
|
||
1) "127.0.0.1"
|
||
2) "6377"
|
||
</code></pre>
|
||
<p>相关语法:</p>
|
||
<pre><code>sentinel get-master-addr-by-name master-name
|
||
</code></pre>
|
||
<h4><strong>查询从节点的信息</strong></h4>
|
||
<pre><code class="language-shell">127.0.0.1:26379> sentinel slaves mymaster #获取方式一
|
||
1) 1) "name"
|
||
2) "127.0.0.1:6379"
|
||
3) "ip"
|
||
4) "127.0.0.1"
|
||
5) "port"
|
||
6) "6379"
|
||
7) "runid"
|
||
8) "14734d6065d745d89f115ca4735e7eeeeaa1a59b"
|
||
9) "flags"
|
||
10) "slave"
|
||
11) "link-pending-commands"
|
||
12) "0"
|
||
13) "link-refcount"
|
||
14) "1"
|
||
15) "last-ping-sent"
|
||
16) "0"
|
||
17) "last-ok-ping-reply"
|
||
18) "389"
|
||
19) "last-ping-reply"
|
||
20) "389"
|
||
21) "down-after-milliseconds"
|
||
22) "30000"
|
||
23) "info-refresh"
|
||
24) "390"
|
||
25) "role-reported"
|
||
26) "slave"
|
||
27) "role-reported-time"
|
||
28) "982798"
|
||
29) "master-link-down-time"
|
||
30) "1582192784000"
|
||
31) "master-link-status"
|
||
32) "err"
|
||
33) "master-host"
|
||
34) "127.0.0.1"
|
||
35) "master-port"
|
||
36) "6377"
|
||
37) "slave-priority"
|
||
38) "100"
|
||
39) "slave-repl-offset"
|
||
40) "1"
|
||
2) 1) "name"
|
||
2) "127.0.0.1:6378"
|
||
3) "ip"
|
||
4) "127.0.0.1"
|
||
5) "port"
|
||
6) "6378"
|
||
7) "runid"
|
||
8) "f9d69479ace6c9eb4a6dffa58ebc1ddf3de456e0"
|
||
9) "flags"
|
||
10) "slave"
|
||
11) "link-pending-commands"
|
||
12) "0"
|
||
13) "link-refcount"
|
||
14) "1"
|
||
15) "last-ping-sent"
|
||
16) "0"
|
||
17) "last-ok-ping-reply"
|
||
18) "390"
|
||
19) "last-ping-reply"
|
||
20) "390"
|
||
21) "down-after-milliseconds"
|
||
22) "30000"
|
||
23) "info-refresh"
|
||
24) "4004"
|
||
25) "role-reported"
|
||
26) "slave"
|
||
27) "role-reported-time"
|
||
28) "76212633"
|
||
29) "master-link-down-time"
|
||
30) "0"
|
||
31) "master-link-status"
|
||
32) "ok"
|
||
33) "master-host"
|
||
34) "127.0.0.1"
|
||
35) "master-port"
|
||
36) "6377"
|
||
37) "slave-priority"
|
||
38) "100"
|
||
39) "slave-repl-offset"
|
||
40) "10811245"
|
||
127.0.0.1:26379> sentinel replicas mymaster #获取方式二
|
||
1) 1) "name"
|
||
2) "127.0.0.1:6379"
|
||
3) "ip"
|
||
4) "127.0.0.1"
|
||
5) "port"
|
||
6) "6379"
|
||
7) "runid"
|
||
8) "14734d6065d745d89f115ca4735e7eeeeaa1a59b"
|
||
9) "flags"
|
||
10) "slave"
|
||
11) "link-pending-commands"
|
||
12) "0"
|
||
13) "link-refcount"
|
||
14) "1"
|
||
15) "last-ping-sent"
|
||
16) "0"
|
||
17) "last-ok-ping-reply"
|
||
18) "100"
|
||
19) "last-ping-reply"
|
||
20) "100"
|
||
21) "down-after-milliseconds"
|
||
22) "30000"
|
||
23) "info-refresh"
|
||
24) "100"
|
||
25) "role-reported"
|
||
26) "slave"
|
||
27) "role-reported-time"
|
||
28) "1071687"
|
||
29) "master-link-down-time"
|
||
30) "1582192873000"
|
||
31) "master-link-status"
|
||
32) "err"
|
||
33) "master-host"
|
||
34) "127.0.0.1"
|
||
35) "master-port"
|
||
36) "6377"
|
||
37) "slave-priority"
|
||
38) "100"
|
||
39) "slave-repl-offset"
|
||
40) "1"
|
||
2) 1) "name"
|
||
2) "127.0.0.1:6378"
|
||
3) "ip"
|
||
4) "127.0.0.1"
|
||
5) "port"
|
||
6) "6378"
|
||
7) "runid"
|
||
8) "f9d69479ace6c9eb4a6dffa58ebc1ddf3de456e0"
|
||
9) "flags"
|
||
10) "slave"
|
||
11) "link-pending-commands"
|
||
12) "0"
|
||
13) "link-refcount"
|
||
14) "1"
|
||
15) "last-ping-sent"
|
||
16) "0"
|
||
17) "last-ok-ping-reply"
|
||
18) "100"
|
||
19) "last-ping-reply"
|
||
20) "100"
|
||
21) "down-after-milliseconds"
|
||
22) "30000"
|
||
23) "info-refresh"
|
||
24) "2496"
|
||
25) "role-reported"
|
||
26) "slave"
|
||
27) "role-reported-time"
|
||
28) "76301522"
|
||
29) "master-link-down-time"
|
||
30) "0"
|
||
31) "master-link-status"
|
||
32) "ok"
|
||
33) "master-host"
|
||
34) "127.0.0.1"
|
||
35) "master-port"
|
||
36) "6377"
|
||
37) "slave-priority"
|
||
38) "100"
|
||
39) "slave-repl-offset"
|
||
40) "10823208"
|
||
</code></pre>
|
||
<p>相关语法:</p>
|
||
<pre><code>sentinel replicas mymaster
|
||
</code></pre>
|
||
<p>或</p>
|
||
<pre><code>sentinel slaves master-name
|
||
</code></pre>
|
||
<h4><strong>查询 Sentinel 集群中的其他 Sentinel 信息</strong></h4>
|
||
<pre><code class="language-shell">127.0.0.1:26379> sentinel sentinels mymaster
|
||
1) 1) "name"
|
||
2) "6455f2f74614a71ce0a63398b2e48d6cd1cf0d06"
|
||
3) "ip"
|
||
4) "127.0.0.1"
|
||
5) "port"
|
||
6) "26377"
|
||
7) "runid"
|
||
8) "6455f2f74614a71ce0a63398b2e48d6cd1cf0d06"
|
||
9) "flags"
|
||
10) "sentinel"
|
||
11) "link-pending-commands"
|
||
12) "0"
|
||
13) "link-refcount"
|
||
14) "1"
|
||
15) "last-ping-sent"
|
||
16) "0"
|
||
17) "last-ok-ping-reply"
|
||
18) "571"
|
||
19) "last-ping-reply"
|
||
20) "571"
|
||
21) "down-after-milliseconds"
|
||
22) "30000"
|
||
23) "last-hello-message"
|
||
24) "1043"
|
||
25) "voted-leader"
|
||
26) "?"
|
||
27) "voted-leader-epoch"
|
||
28) "0"
|
||
</code></pre>
|
||
<p>相关语法:</p>
|
||
<pre><code>sentinel sentinels master-name
|
||
</code></pre>
|
||
<h4><strong>检查可用 Sentinel 的数量</strong></h4>
|
||
<pre><code class="language-shell">127.0.0.1:26379> sentinel ckquorum mymaster
|
||
OK 2 usable Sentinels. Quorum and failover authorization can be reached
|
||
</code></pre>
|
||
<p>有两个可用的 Sentinel,可用完成仲裁和故障转移授权。</p>
|
||
<p>相关语法:</p>
|
||
<pre><code>sentinel ckquorum master-name
|
||
</code></pre>
|
||
<h4><strong>强制故障转移</strong></h4>
|
||
<pre><code class="language-shell">127.0.0.1:26379> sentinel failover mymaster
|
||
OK
|
||
</code></pre>
|
||
<p>相关语法:</p>
|
||
<pre><code>sentinel failover master-name
|
||
</code></pre>
|
||
<h3>在线修改配置信息</h3>
|
||
<p>在 Redis 2.8.4 之前如果需要修改 Sentinel 的配置文件,例如添加或删除一个监视主节点,需要先停止 Sentinel 服务,再找到配置文件修改之后,重新启动 Sentinel 才行,这样就给我们带来了很多的不便,尤其是生产环境的 Sentinel,正常情况下如果是非致命问题我们是不能手动停止服务的,幸运的是 Redis 2.8.4 之后,我们可以不停机在线修改配置文件了,修改命令有以下几个。</p>
|
||
<h4><strong>增加监视主节点</strong></h4>
|
||
<p>使用 <code>sentinel monitor mymaster IP Port Quorum</code> 命令来添加监视主节点,如下命令所示:</p>
|
||
<pre><code class="language-shell">127.0.0.1:26379> sentinel monitor mymaster 127.0.0.1 6379 2
|
||
OK
|
||
</code></pre>
|
||
<p>OK 表示添加监视主节点成功。</p>
|
||
<h4><strong>移除主节点的监视</strong></h4>
|
||
<p>使用 <code>sentinel remove master-name</code> 命令来实现移除主节点的监视,如下命令所示:</p>
|
||
<pre><code class="language-shell">127.0.0.1:26379> sentinel remove mymaster
|
||
OK
|
||
</code></pre>
|
||
<p>OK 表示操作成功。</p>
|
||
<h4><strong>修改 quorum 参数</strong></h4>
|
||
<p>使用 <code>sentinel set master-name quorum n</code> 来修改 quorum 参数,如下命令所示:</p>
|
||
<pre><code class="language-shell">127.0.0.1:26379> sentinel set mymaster quorum 1
|
||
OK
|
||
</code></pre>
|
||
<p>quorum 参数用来表示确认主节点下线的 Sentinel 数量,如果 quorum 设置为 1 表示只要有一台 Sentinel 确认主观下线后,这个主节点就客观(真正地)下线了。</p>
|
||
<blockquote>
|
||
<p>小贴士:以上所有对配置文件的修改,都会自动被刷新到物理配置文件 sentinel.conf 中。</p>
|
||
</blockquote>
|
||
<h3>代码实战</h3>
|
||
<p>本文我们通过 Java 代码来实现,通过 Sentinel 连接信息获取相关 Redis 客户端,再进行相关 Redis 操作,这样 Sentinel 就会帮我们做容灾恢复,我们就不用担心操作某一个 Redis 服务器端,因为服务器挂了之后就会导致程序不可用了,具体实现代码如下:</p>
|
||
<pre><code class="language-java">import redis.clients.jedis.Jedis;
|
||
import redis.clients.jedis.JedisSentinelPool;
|
||
import utils.Config;
|
||
import java.util.HashSet;
|
||
import java.util.Set;
|
||
public class SentinelExample {
|
||
// master name
|
||
private static String _MASTER_NAME = "mymaster";
|
||
public static void main(String[] args) {
|
||
// Sentinel 配置信息
|
||
Set<String> set = new HashSet<>();
|
||
// 连接信息 ip:port
|
||
set.add("127.0.0.1:26379");
|
||
// 创建 Sentinel 连接池
|
||
JedisSentinelPool jedisSentinel = new JedisSentinelPool(_MASTER_NAME,
|
||
set, Config.REDIS_AUTH);
|
||
// 获取 Redis 客户端
|
||
Jedis jedis = jedisSentinel.getResource();
|
||
// 设置元素
|
||
String setRes = jedis.set("key", "Hello, redis.");
|
||
System.out.println(setRes);
|
||
// 获取元素
|
||
System.out.println(jedis.get("key"));
|
||
}
|
||
}
|
||
</code></pre>
|
||
<p>以上程序执行结果如下:</p>
|
||
<pre><code>OK
|
||
Hello, redis.
|
||
</code></pre>
|
||
<h3>小结</h3>
|
||
<p>本文我们讲了 Sentinel 相关的命令操作,主要是用于查询相关主从节点和其他 Sentinel 信息的,还可以执行强制故障转移等,我们还讲了 2.8.4 提供的在线修改 Sentinel 参数的三个方法,方便我们更好的使用 Sentinel,最后用代码实现了通过 Sentinel 获取主节点并进行 Redis 服务器操作的实例,这样就讲完整个 Sentinel 的介绍和应用。</p>
|
||
</div>
|
||
</div>
|
||
<div>
|
||
<div style="float: left">
|
||
<a href="/专栏/Redis 核心原理与实战/37 实战:Redis哨兵模式(上).md.html">上一页</a>
|
||
</div>
|
||
<div style="float: right">
|
||
<a href="/专栏/Redis 核心原理与实战/39 实战:Redis 集群模式(上).md.html">下一页</a>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<a class="off-canvas-overlay" onclick="hide_canvas()"></a>
|
||
</div>
|
||
<script defer src="https://static.cloudflareinsights.com/beacon.min.js/v652eace1692a40cfa3763df669d7439c1639079717194" integrity="sha512-Gi7xpJR8tSkrpF7aordPZQlW2DLtzUlZcumS8dMQjwDHEnw9I7ZLyiOj/6tZStRBGtGgN6ceN6cMH8z7etPGlw==" data-cf-beacon='{"rayId":"70997403ace33d60","version":"2021.12.0","r":1,"token":"1f5d475227ce4f0089a7cff1ab17c0f5","si":100}' crossorigin="anonymous"></script>
|
||
</body>
|
||
<!-- Global site tag (gtag.js) - Google Analytics -->
|
||
<script async src="https://www.googletagmanager.com/gtag/js?id=G-NPSEEVD756"></script>
|
||
<script>
|
||
window.dataLayer = window.dataLayer || [];
|
||
function gtag() {
|
||
dataLayer.push(arguments);
|
||
}
|
||
gtag('js', new Date());
|
||
gtag('config', 'G-NPSEEVD756');
|
||
var path = window.location.pathname
|
||
var cookie = getCookie("lastPath");
|
||
console.log(path)
|
||
if (path.replace("/", "") === "") {
|
||
if (cookie.replace("/", "") !== "") {
|
||
console.log(cookie)
|
||
document.getElementById("tip").innerHTML = "<a href='" + cookie + "'>跳转到上次进度</a>"
|
||
}
|
||
} else {
|
||
setCookie("lastPath", path)
|
||
}
|
||
function setCookie(cname, cvalue) {
|
||
var d = new Date();
|
||
d.setTime(d.getTime() + (180 * 24 * 60 * 60 * 1000));
|
||
var expires = "expires=" + d.toGMTString();
|
||
document.cookie = cname + "=" + cvalue + "; " + expires + ";path = /";
|
||
}
|
||
function getCookie(cname) {
|
||
var name = cname + "=";
|
||
var ca = document.cookie.split(';');
|
||
for (var i = 0; i < ca.length; i++) {
|
||
var c = ca[i].trim();
|
||
if (c.indexOf(name) === 0) return c.substring(name.length, c.length);
|
||
}
|
||
return "";
|
||
}
|
||
</script>
|
||
</html>
|