简单回顾下RDB文件的建立。html
有两种建立方式:java
在redis服务启动时,若是检测到RDB文件,会进行自动载入。git
若是开启了AOF,则会优先AOFgithub
save 900 1
save 300 10
save 60 10000
复制代码
这是redis.conf
配置文件中关于RDB save时机的配置,它映射在redisServer
结构体的saveparams
字段中:redis
struct redisServer{
....
// 保存了redis.conf配置的属性
struct saveparam *saveparams;
// 记录上一次save的时间
time_t lastsave;
// 修改计数器
long long dirty;
...
};
复制代码
那来看看它怎么保存的:数据库
struct saveparam {
// 秒数
time_t seconds;
// 修改次数
int changes;
};
复制代码
redis本身有一个定时任务每100毫秒执行一次,其中有一个任务就是检查save条件是否知足,如何判断的呢?就是用lastsave
与saveparam.seconds
比较时间是否知足,dirty
与changes
比较修改次数是否知足。bash
那bgsave如何实现呢,new一个子线程,而后拷贝个数据副本,而后和save同样处理。服务器
好了,到这里,用Java写一个这应该是没问题了,那RDB的文件结构如何设计呢? 咱们来看看redis的设计。app
REDIS+数据库版本号+数据类型+数据+EOF(表示数据结束)(377)+检验和性能
咱们知道java中Class文件结构很复杂,由于它包含了常量、接口、类、父类、字段等面向对象的信息,而RDB的就比较简单了,由于它只须要存放数据便可。
和class结构同样,它的开头也是文件标识REDIS
+版本号标识.
[root@izuf6i2jk9azj2te13kjx8z redis-4.0.9]# od -c dump.rdb
0000000 R E D I S 0 0 0 8 372 \t r e d i s
0000020 - v e r 005 4 . 0 . 9 372 \n r e d i
0000040 s - b i t s 300 @ 372 005 c t i m e 302
0000060 231 ; 017 ] 372 \b u s e d - m e m 302 310
0000100 p \r \0 372 \f a o f - p r e a m b l
0000120 e 300 \0 376 \0 373 ( \0 \0 006 k - 7 5 9 9
0000140 006 v - 7 5 9 9 \0 022 c p t : 254 355 \0
0000160 005 t \0 \a g e t O n e 4 303 L 220 ^ 303
0000200 037 254 355 \0 005 s r \0 % c o m . f a n
0000220 t . c o r e . r e s p o n s e .
0000240 S 005 e r v e r R 240 016 030 222 224 e 250 :
0000260 035 323 ? 002 \0 003 I \0 006 s t a t u s L
0000300 \0 004 d a \t 031 \0 022 L j a v a / l
0000320 a n g / O b j e c t ; L \0 003 m s
0000340 g 340 005 032 \f S t r i n g ; x p \0 \0
0000360 \0 310 y \0 036 340 005 y 037 p o j o . C
0000400 o m p e t i t i o n Z 276 231 334 b 025
...
0140540 \a 004 j a v a 377 \v 006 n u m b e r 024
0140560 002 \0 \0 \0 006 \0 \0 \0 001 \0 002 \0 003 \0 004 \0
0140600 005 \0 006 \0 016 004 l i s t 001 027 027 \0 \0 \0
0140620 024 \0 \0 \0 006 \0 \0 362 002 363 002 364 002 365 002 366
0140640 002 367 377 377 - 022 036 ] 367 332 257 _
复制代码
分析:
R E D I S:RDB文件标志
0 0 0 8:版本号
372:结束符
r e d i s
0000020 - v e r 005 4 . 0 . 9:redis-version4.0.9
r e d i
0000040 s - b i t s 300 @:redis的位数64或32
c t i m e 302 0000060 231 ; 017 ]:时间戳
u s e d - m e m 302 310 0000100 p \r \0:redis使用内存的大小
374:RDB_OPCODE_EXPIRETIME_MS(带有过时时间标识)
\0: 表示字符串
最后8字节为校验和
复制代码
更详细的能够查看http://redisbook.com/preview/rdb/rdb_struct.html
手写过Jedis的朋友都熟悉RESP协议,RDB的数据段和它的排版方式很类似。 好比:\0 \0 003 m s g 005 h e l l o 377
就表示键值对:msg(3个长度):hello(5个长度)
AOF以拼接和重写命令的方式来实现。
# 是否开启aof
appendonly yes
# 文件名称
appendfilename "appendonly.aof"
# 同步方式
##每次收到写命令就当即强制写入磁盘,最慢的,可是保证彻底的持久化,不推荐使用
# appendfsync always
##每秒钟强制写入磁盘一次,在性能和持久化方面作了很好的折中,系统默认
appendfsync everysec
##彻底依赖os,性能最好,持久化没保证
# appendfsync no
# aof重写期间是否同步
no-appendfsync-on-rewrite no
# 重写触发配置
auto-aof-rewrite-percentage 100
auto-aof-rewrite-min-size 64mb
# 加载aof时若是有错如何处理
aof-load-truncated yes
# 文件重写策略
aof-rewrite-incremental-fsync yes
复制代码
这一段配置中,你们着重理解同步方式的配置。redis默认采用的每秒一次写入AOF文件的策略。
struct redisServer {
// ...
// 存放AOF缓冲
sds aof_buf;
// ...
};
复制代码
当有新的命令进来,redis就会将其(协议化后)追加到aof_buf
的末尾。
同理,redis的事件循环也会监听AOF的配置,若是知足配置文件中的同步方式appendfsync everysec等
,就会将aof_buf
中的内容保存到AOF文件里。
咱们知道,redis对AOF有重写机制,用来控制AOF文件的大小。
发生在重写列表、哈希表、集合、有序集合可能会带有多个元素的键时。
不是,若是它的值超过64项,则会用多条命令来完成。(避免客户端输入缓冲区溢出)
Redis不但愿AOF重写形成服务器阻塞,因此用子进程(带有数据副本)去处理。
不会。为了解决这个问题,Reids设置了AOF重写缓冲区(建立子进程后开启),当Redis执行命令时,redis会同时将这个信息发送给aof_buf
和AOF重写缓冲区。
Redis采用的是惰性删除和按期删除,配合这两种策略来取得CPU和内存的平衡。
不包含。
在生成RDB和AOF文件时,程序会对键进行检查,已过时的键不保存到文件中。
关注个人公众号,随时阅读个人所有文章。
想看往期文章, 请点击个人GitHub地址: github.com/fantj2016/j…