前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >聊聊PostgreSQL的Replication

聊聊PostgreSQL的Replication

作者头像
用户4700054
发布于 2022-08-17 04:42:49
发布于 2022-08-17 04:42:49
1.6K00
代码可运行
举报
运行总次数:0
代码可运行

CAP理论

  • consistency:在整个集群角度来看,每个节点是看到的数据一致的;不能出现集群中节点出现数据不一致的问题
  • vailability:集群中节点,只有有一个节点能提供服务
  • partitioning:集群中的节点之间网络出现问题,造成集群中一部分节点和另外一部分节点互相无法访问

基本术语

  • Master节点:提供数据写的服务节点
  • Standby节点:根据主节点(master节点)数据更改,这些更改同步到另外一个节点(standby节点)
  • Warm Standby节点:可以提升为master节点的standby节点
  • Hot Standby节点:主要提供读服务的standby节点
PostgreSQL支持的Replication方案
  • 基于文件或者磁盘Replication:这种方式采用共享磁盘或者共享NAS方式,采用了存储计算分离的方式,如果采用这样的方式,PostgreSQL是计算节点,底层的是一个分布式块存储或者分布式文件存储。这样的好处很明显,只需要保证计算层的高可用即可,但是弊端也很明显由于底层是分布式存储,PG性能取决于分布式文件存储。如果底层的分布式存储做的足够健壮,数据基本不会丢失
  • 基于wal的物理Replication:postgresql支持物理复制,其原理的就是先把Master节点的热备,然后传输到standby节点,在standby节点恢复;最后master不断的发送数据变更wal日志给standby节点,standby节点不断的接受wal日志,然后进行apply。物理复制是针对所有的Master节点上的databaase.由于wal是基于page的级别的,standby节点应用比较快,开销小。在物理复制中,Master节点会运行多个wal send进程;Standby节点会运行多个wal recv进程和startup进程,send是master发送wal日志的进程;recv进程是standby节点接受wal日志的进程,startup进程是standby节点apply wal日志的进程。
  • 基于SQL的逻辑Replication:基本原理是应用端发出更改请求,master不断的产生日志,紧接着master的send进程读取wal日志,然后经过decode模块进行解析wal日志转换为类似于sql的方式发送给standby的recv进程,recv进程接受到sql日志,发送给standby的execute模块进行解码成为sql语句,然后执行sql语句,产生wal日志。
Replication实践
物理复制
  • 准备两个PG实例
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
// 主节点 ip=127.0.0.1,port = 5432
// 从节点 ip=127.0.0.1,port = 5433
[perrynzhou@CentOS8-Dev /postgres]$ ps -ef|grep -v grep|grep postgre
perrynz+   13955       1  0 15:22 ?        00:00:00 /usr/local/postgres/bin/postgres -D /postgres/data1
perrynz+   13957   13955  0 15:22 ?        00:00:00 postgres: checkpointer 
perrynz+   13958   13955  0 15:22 ?        00:00:00 postgres: background writer 
perrynz+   13959   13955  0 15:22 ?        00:00:00 postgres: walwriter 
perrynz+   13960   13955  0 15:22 ?        00:00:00 postgres: autovacuum launcher 
perrynz+   13961   13955  0 15:22 ?        00:00:00 postgres: archiver 
perrynz+   13962   13955  0 15:22 ?        00:00:00 postgres: stats collector 
perrynz+   13963   13955  0 15:22 ?        00:00:00 postgres: logical replication launcher 
perrynz+   13966       1  0 15:22 ?        00:00:00 /usr/local/postgres/bin/postgres -D /postgres/data2
perrynz+   13968   13966  0 15:22 ?        00:00:00 postgres: checkpointer 
perrynz+   13969   13966  0 15:22 ?        00:00:00 postgres: background writer 
perrynz+   13970   13966  0 15:22 ?        00:00:00 postgres: walwriter 
perrynz+   13971   13966  0 15:22 ?        00:00:00 postgres: autovacuum launcher 
perrynz+   13972   13966  0 15:22 ?        00:00:00 postgres: stats collector 
perrynz+   13973   13966  0 15:22 ?        00:00:00 postgres: logical replication launcher
  • 在主节点创建复制账户和备份主节点
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
// 主库创建数据库用户
$ /usr/local/postgres/bin/psql  -h 127.0.0.1 postgres -p 5432 -U perrynzhou 
psql (14rc1)
Type "help" for help.

// 复制槽很重要,防止主库过早清理Wal
// 删除槽位信息 select pg_drop_replication_slot('slot_name');

postgres=# select * from pg_create_physical_replication_slot('db1_repl_slot');
   slot_name   | lsn 
---------------+-----
 db1_repl_slot | 
(1 row)

postgres=# select slot_name, slot_type, active, wal_status from pg_replication_slots;
   slot_name   | slot_type | active | wal_status 
---------------+-----------+--------+------------
 db1_repl_slot | physical  | f      | 
(1 row)


// 备份主库
$ /usr/local/postgres/bin/pg_basebackup --pgdata /postgres/master_backup --format=p \
   --write-recovery-conf --checkpoint=fast --label=mffb --progress \
   --host=127.0.0.1  --port=5432 --username=perrynzhou
166886/166886 kB (100%), 1/1 tablespace

// 停止从库
/usr/local/postgres/bin/pg_ctl -D /postgres/data2/ -l pg_logfile2 stop
// 删除从库数据库
rm -rf /postgres/data2 && mv /postgres/master_backup  /postgres/data2

 // 添加配置到从库的postgresql.conf
primary_conninfo = 'host=127.0.0.1 port=5432 user=perrynzhou password=zhoulin'
primary_slot_name = 'db1_repl_slot'
  • 主从配置
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
// 主库 postgresql.conf
port = 5432                             
max_connections = 100                   
shared_buffers = 128MB                 
dynamic_shared_memory_type = posix     
wal_level = replica
max_wal_size = 1GB
min_wal_size = 80MB
archive_mode = on
archive_command = 'cp %p /postgres/archive1/%f '
listen_addresses = '*'

//从库postgresql.conf
listen_addresses = '*'
archive_command = 'cp %p /postgres/archive2/%f '
port = 5433                             
max_connections = 100                
shared_buffers = 128MB                  
dynamic_shared_memory_type = posix     
max_wal_size = 1GB
min_wal_size = 80MB
wal_level = replica
hot_standby = on
max_standby_streaming_delay = 30s
hot_standby_feedback = on
primary_conninfo = 'host=127.0.0.1 port=5432 user=perrynzhou password=zhoulin'
primary_slot_name = 'db1_repl_slot'
  • 主从验证
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
// 主库
$ /usr/local/postgres/bin/psql  -h 127.0.0.1 postgres -p 5432 -U perrynzhou
psql (14rc1)

postgres=# create table tt1(id int);
CREATE TABLE
postgres=# insert into tt1 values(1);
INSERT 0 1
postgres=#

// 从库
$ /usr/local/postgres/bin/psql  -h 127.0.0.1 postgres -p 5433 -U perrynzhou
psql (14rc1)

postgres=# \d
         List of relations
 Schema | Name | Type  |   Owner    
--------+------+-------+------------
 public | tt1  | table | perrynzhou
(1 row)

postgres=# \t
Tuples only is on.
postgres=# select * from tt1;
  1

postgres=#
逻辑复制
  • 主从配置
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
port = 5432                             
max_connections = 100                   
shared_buffers = 128MB                 
dynamic_shared_memory_type = posix     
wal_level = logical
max_wal_size = 1GB
min_wal_size = 80MB
archive_mode = on
archive_command = 'cp %p /postgres/archive1/%f '
listen_addresses = '*'

//从库postgresql.conf
listen_addresses = '*'
archive_command = 'cp %p /postgres/archive2/%f '
port = 5433                             
max_connections = 100                
shared_buffers = 128MB                  
dynamic_shared_memory_type = posix     
max_wal_size = 1GB
min_wal_size = 80MB
wal_level = replica
hot_standby = on
max_standby_streaming_delay = 30s
hot_standby_feedback = on
primary_conninfo = 'host=127.0.0.1 port=5432 user=perrynzhou password=zhoulin'
primary_slot_name = 'db1_repl_slot'
  • 配置主库和从库
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
// 配置主库
$ /usr/local/postgres/bin/psql  -h 127.0.0.1 postgres -p 5432 -U perrynzhou
psql (14rc1)

postgres=# CREATE PUBLICATION my_publication FOR ALL TABLES;
CREATE PUBLICATION
postgres=# 

// 配置从库
$ /usr/local/postgres/bin/psql  -h 127.0.0.1 postgres -p 5433 -U perrynzhou
psql (14rc1)

postgres=# CREATE SUBSCRIPTION my_subscription CONNECTION 'host=127.0.0.1 port=5432 user=perrynzhou dbname=postgres' PUBLICATION my_publication WITH (copy_data=false);
NOTICE:  created replication slot "my_subscription" on publisher
CREATE SUBSCRIPTION
  • 验证
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
// 主库插入数据
$ /usr/local/postgres/bin/psql  -h 127.0.0.1 postgres -p 5432 -U perrynzhou
postgres=# select * from tt2;
 id 
----
(0 rows)

postgres=# insert into tt2 values(100);
INSERT 0 1

// 从库查看数据
[perrynzhou@CentOS8-Dev /postgres]$ /usr/local/postgres/bin/psql  -h 127.0.0.1 postgres -p 5433 -U perrynzhou
postgres=# select * from tt2;
 id  
-----
 100
(1 row)

postgres=# 

// 主库查看复制槽位信息
postgres=# select slot_name, slot_type, active, wal_status from pg_replication_slots;
    slot_name    | slot_type | active | wal_status 
-----------------+-----------+--------+------------
 my_subscription | logical   | t      | reserved
(2 rows)
本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2021-12-25,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 存储内核技术交流 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
Python基础(二)
原文首发于2019-04-15:https://maoli.blog.csdn.net/article/details/89315948
润森
2020/04/08
4020
Python基础(二)
dict 和 set 的 15 个经典使用例子
公众号原文:https://mp.weixin.qq.com/s/WwJdPebK_hFocS1lk7MWhA
AI悦创
2022/05/11
4290
PEP 584:字典合并操作符来了
花下猫语:最新发布的 Python 3.9 预览版合入了一个很小的改动(PEP-584),关于这个特性本身不需要多说,只需要一两个示例,大家就能接受使用。但是,就像我之前介绍过的一些 PEP 一样,关于它的来龙去脉和引起的相关讨论,都是挺有意思的细节。今天分享的文章,对此有详尽的梳理,推荐大家一读。
Python猫
2020/03/03
1.3K0
Python进阶8——字典与散列表,字符串编解码
Python用散列表来实现字典,散列表就是稀疏数组(数组中有空白元素),散列表中的元素叫做表元,字典的每个键值对都占用一个表元,一个表元分成两个部分,一个是对键的应用,另一个是对值的引用,因为表元的大小一致,所以可以通过稀疏数组(散列表)的偏移量读取指定的表元
用户7886150
2020/12/30
1.4K0
python3之dict字典
## d.setdefault( x, [ , y ] )  返回字典 d 中键 x 对应的值,若键 x 不存在,则返回y, 并将 x : y 作为键值对添加到字典中,y 的默认值为 None
py3study
2020/01/03
6270
Python中相见恨晚的技巧
词典对象前面的双星号可以让你把该词典的内容作为命名参数输入到函数中。词典的秘钥是参数名,值是传递给函数的值。你甚至不需要称它为 kwargs!
幻影龙王
2021/08/22
6701
Python中相见恨晚的技巧
由一个简单的Python合并字典问题引发的思考,如何优化我们的代码?
今天我们的题目是《由一个简单的Python合并字典问题引发的思考,如何优化我们的代码?》,为什么会有这个话题呢?起因是今天和一位刚刚面试完Python开发岗位的朋友交流,这个问题也是他在面试中遇到的问题:
云爬虫技术研究笔记
2019/11/15
1.5K0
Python 3.9 你所需要知道的都在这里
# 只有一行代码,看上去很酷,缺点是这种hack技巧只有在字典的键是字符串时才有效。
用户7886150
2021/01/12
4640
Python中相见恨晚的技巧
词典对象前面的双星号可以让你把该词典的内容作为命名参数输入到函数中。词典的秘钥是参数名,值是传递给函数的值。你甚至不需要称它为 kwargs!
Qwe7
2022/02/02
3040
python元组-字典-集合及其内置方法(下)
  列表反向、排序   reverse   sort(带参数,可指定按正序/ 反序排)
suwanbin
2019/09/26
1.2K0
python学习_18
字典 字典是无序的 字典的key只能是不可变对象,不能是list dict 创建字典 创建空字典,并赋值
py3study
2020/01/15
4390
8个Python小技巧
isinstance 函数可用于判断实例的类型,其实它的第二个参数可以是多个数据类型组成的元组
胡八万
2022/05/16
3150
python 基础知识第8讲:序列之字典
删除之后他会将删除的key-value 作为返回值返回 返回值是一个元组,元组中有2个元素 第一个是删除的Key 第二个是删除的value
小海怪的互联网
2019/08/23
7250
Python | 原来 collections 这么好用!!
https://blog.csdn.net/mall_lucy/article/details/108822795
咸鱼学Python
2020/11/20
9050
Python | 原来 collections 这么好用!!
Python字典使用(八)
字典的每个键值(key=>value)对用冒号分隔,每对之间用逗号分割,整个字典包括花括号在内,如下所示:
py3study
2020/01/10
8090
【每日一记3.16】python学习记录
   列表用【】包含,内有数据对象,每个数据对象以‘,’分隔,每个数据对象称为元素
py3study
2020/01/07
6880
Python中字典的详细用法
#字典 #字典是Python中唯一内建的映射类型。字典中没有特殊的顺序,但都是存储在一个特定的键(key)下面,键可以是数字,字符串,甚至是元组 #一、字典的使用 #在某些情况下,字典比列表更加适用: #1、表示一个游戏棋盘的状态,每个键都是由坐标值组成的元组 #2、存储文件修改时间,用文件名作为键; #3、数字电话\地址薄 #1、使用列表创建一个电话本,(这里用字符串表示电话号码,以0开头的数字回会被编译成8进制数字) name=["A","B","C","D"] phone=["2341","910
hankleo
2020/09/17
9540
python字典和集合
1. 字典字典是python中唯一的映射类型,采用键值对(key-value)的形式存储数据。python对key进行哈希函数运算,根据计算的结果决定value的存储地址,所以字典是无序存储的,且key必须是可哈希的。可哈希表示key必须是不可变类型,如:数字、字符串、只含不可变类型元素的元组(1,2,3,’abc’)、实现__hash__()方法的自定义对象(因为__hash__()须返回一个整数,否则会出现异常:TypeError: an integer is required)。可以用hash(obj
py3study
2020/01/07
5720
Python字典方法
2、copy 方法copy返回一个新字典,其包含的键值对与原来的字典相同(这各方法是浅复制,因为值本身是原件,而非副本)
王大力测试进阶之路
2022/06/07
3790
python︱ collections模块(namedtuple/defaultdict/OrderedDict等)
给[‘x’,‘y’]这个tuple命名为point,这个tuple中,第一个空位命名为'x',第二个为'y'。
悟乙己
2019/05/26
1.9K0
相关推荐
Python基础(二)
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验