36.Redis核心设计原理

devtools/2024/11/14 15:10:31/

本文针对前面的讲解做一次总结

1.Redis基本特性

    1.非关系型键值对数据库,可以根据键以O(1)的时间复杂度取出或插入关联值

    2.Redis的数据是存在内存中的

    3.键值对中键的类型可以是字符串,整型,浮点型等,且键是唯一的

    4.键值对中的值类型可以是字符串,哈希,列表,集合,排序集合等

    5.Redis内置了复制,磁盘持久化,LUA脚本,事务,SSL,ACL,客户端缓存,客户端代理等功能

    6.通过Redis哨兵和Redis集群模式提供高可用性

2.Redis应用场景

2.1 缓存

可以作为缓存使用,数据存在内存中,数据存取效率非常高

2.2 计数器

实现计数器功能。Redis 这种内存型数据库的读写性能非常高可以对 String 进行自增自减运算,很适合存储频繁读写的计数量。

2.3 分布式ID生成

利用自增特性一次请求一个大一点的步长如 incr 2000,缓存在本地使用,用完再请求。

2.4 海量数据统计

位图_(bitmap):存储是否参过某次活动,是否已读谋篇文章,用户是否为会员,日活统计

2.5 会话缓存

可以使用 Redis 来统一存储多台应用服务器的会话信息。当应用服务器不再存储用户的会话信息,也就不再具有状态,一个用户可以请求任意一个应用服务器,从而更容易实现高可用性以及可伸缩性。

2.6 分布式队列/阻塞队列

List 是一个双向链表,可以通过 lpush/rpush 和 rpop/lpop 写入和读取消息。可以通过使用brpop/blpop 来实现阻塞队列。

2.7 分布式锁实现

在分布式场景下,无法使用基于进程的锁来对多个节点上的进程进行同步。可以使用 Redis 自带的SETNX 命令实现分布式锁。

2.8 热点数据存储

最新评论,最新文章列表,使用list 存储,ltrim取出热点数据,删除老数据。

2.9 社交类需求

Set 可以实现交集,从而实现共同好友等功能,Set通过求差集,可以进行好友推荐,文章推荐。

2.10 排行榜

sorted set可以实现有序性操作,从而实现排行榜等功能。

2.11 延迟队列

使用sorted set(zset),使用 【当前时间戳 +需要延迟的时长】做score,消息内容作为元素,调用zadd来生产消息,消费者使用zrangbyscore获取当前时间之前的数据做轮询处理。消费完再删除任务 rem keymember

3.Redis键(Key)存储

所有的键都是String类型,都存储在buf里面,参考下面DB存储结构

4.Redis(Value)DB存储

redis的数据结构更偏向于map,在redis更具象化叫做dict,通过key找到value,存储海量数据;存储结构包括

1.数组:O(1)

2.链表: O(N)

4.1 String

Redis 3.2以前

    C语言中使用字符串数组存储 char buf[] = "xiehao\0";但是C语言有个问题,在read键时,可能会漏读String键的长度,所以redis重新定义了一个SDS存储方式:

如下以键“xiehao”的键的存储位案例分析;

在 C语言中存储 C:char data[] = "xiehao\0",默认结尾会带一个“\0”;

SDS:

free:0

len:6

char buf [] = "xiehao"

当扩容free长度不够时

将键"xiehao"变成"xiehao123",数组长度会自动计算多一些( 6+3)*2 =18

SDS:

free: 9

len :9

char buf[]="xiehao123"

当扩容free长度够用时

如果再次扩容变成"xiehao123456",则增加了3个长度,判断free剩余长度可以容纳,则直接使用free,不需要扩容

SDS:

free: 6

len :12

char buf[]="xiehao123456"

当扩容到1024 *1024(也就是内存为1M时)则不会在继续扩容

Redis 3.2后

len:长度

buf: 数据

alloc:buf分配的长度

flags:1字节 类型和业务数据长度,根据类型指向定义不同长度的内存;

类型有: sdshrd5 、 sdshrd8、 sdshrd16 、 sdshrd32 、 sdshrd64

使用SDS优点:

1.二进制安全的数据结构;

2.提供了内存预分配机制,避免了频繁的内存分配

3.兼容C语言的函数库

4.2 Hash

    Hash数据结构根据Hash(key)进行随机散列,并且将hash值变成自然数,这样可以将自然数作为索引来定位数据,大大提高效率;若发生hash碰撞,则进行链表关联;

例如:

创建一个大的数组:

arr[4]

hash( key)->自然数[大]%4=「0,arr.size]

1.任意相同的输入,一定能得到相同的输出

2.不同的输入,有可能得到相同的输出

定义3个键值对 (k1,v1)(k2,v2),(k3,v3)

hash(k1)%4=0

hash(k2)%4=1

hash(k3)%4=1

arr[0]->(k1,v1,next->null)

arr[1]->(k3,v3,next ->k2 )(k2,v2,next ->null)

链表 :redis:头插入方式

Redis 有16个DB

typedef struct redisDb{

dict *dict; (k-v)

dict *expires; (过期时间)

dict *blocking keys; (阻塞队列

dict *ready keys; (客户端

dict *watched_keys; (事务处理

int id; (SELECT * 命令 - 数据库索引

long long avg_ttl; (数据统计、遍历等

unsigned long expires_cursor;

list *defrag _later;

}redisDb;

typedef struct dict{

dictType *type; (类型

void *privdata;

dictht ht[2]; (扩容

long rehashidx; (渐进式rehash)

unsigned long iterators;

}dict;

typedef struct dictht{

dictEntry **table;

unsigned long size;

unsigned long sizemask;

unsigned long used;

}dictht;

typedef struct dictEntry{

void *key;

union{

void *val;

uint64t u64;

int64t s64;

double d;

}V;

struct dictEntry *next;

}dictEntry;

typedef struct redisObject{

unsigned type:4; (约束类型

unsigned encoding:4; (数据类型

unsigned Iru:LRU BITS;

int refcount;

void *ptr; (数据存储位置

}robj;

4.3 List

List存储结构采用quickList(双端链表) 和 zipList存储;数据结构K-V(array)

zipList

zibytes: 4字节 存多少数据

zltail:尾节点的索引

zllen:有多少元素

zlend:1字节 数据结尾标识

prerawlen: 前一个数据的信息

len: 当前数据的类型

data:当前的数据

quickList:

head:

tail

count:

length:

quickListNode:节点数据

zl:ziplist的数据

单个ziplist节点最大能存储  8kb  ,超过则进行分裂,将数据存储在新的ziplist节点中

hash

4.4 Set

Set为无序的,自动去重的集合数据类型,Set 数据结构底层实现为一个value为x`的字典(dict),当数据可以用整形表示时,Set集合将被编码为intset数据结构。两个条件任意满足时Set将用hashtable存储数据。K - V(array)

1:元素个数大于set-max-intset-entries;

2:元素无法用整形表示;

typedef struct intset{

uint32t encoding;

uint32t length;

int8t contents[];

}intset;

encoding:编码类型

length:元素个数

contents[]:元素存储

整数集合是一个有序的,存储整型数据的结构。整型集合在Redis中可以保存int16t,int32 t,int64t类型的整型数据,并且可以保证集合中不会出现重复数据。

4.5 Zset

ZSet 为有序的,自动去重的集合数据类型,ZSet 数据结构底层实现为字典(dict)+跳表(skiplist),当数据比较少时,用ziplist编码结构存储。K-V(array)

zipList

zset-max-ziplist-entries 128/元素个数超过128,将用skiplist编码

zset-max-ziplist-value64//单个元素大小超过64 byte,将用skiplist编码

skipList

typedef struct zskiplist{

struct zskiplistNode *header,(头节点,做索引,不存数据)

*t ail; (尾节点,会

存数据)

unsigned long length;(元素个数)

int level; (层高)

}zskiplist;

跳表算法:

数据项:N

index1 : N/2

index2 : N/2^2

index3: N/2^3

indexk: N/2^k

N/2^K = 2

2^K = N/2

k = log2 N/2

k = logN

ZSet  为有序的,自动去重的集合数据类型,ZSet 数据结构底层实现为 字典(dict) + 跳表(skiplist) ,当数据比较少时,用ziplist编码结构存储。


http://www.ppmy.cn/devtools/133682.html

相关文章

lua入门教程:math

在Lua中,math库是一个非常重要的内置库,它提供了许多用于数学计算的函数。这些函数可以处理各种数学运算,包括基本的算术运算、三角函数、对数函数、随机数生成等。结合你之前提到的Lua中的数字遵循IEEE 754双精度浮点标准,我们可…

如何设置el-date-picker的默认截止时间为“23:59:59”

总结网上的方法&#xff0c;最好用的是最后一个 设置的关键是:default-time“[‘00:00:00’, ‘23:59:59’]” <el-date-picker:default-time"[00:00:00, 23:59:59]"v-model"formData.dischargeTime"type"datetimerange"range-separator&quo…

labview连接sql server数据库

通常涉及到庞大的数据量时&#xff0c;我们会优先考虑sql server&#xff0c;他相对存储的数据会多一些&#xff0c;对于最近这个项目刚好可以用得到&#xff0c;下面我们来说一下关于labview连接sql server数据库。 首先是连接信息&#xff0c;和连接ACCESS数据库一样&#x…

1Panel修改PostgreSQL时区

需求 1Panel安装的PostgreSQL默认是UTC时区&#xff0c;需要将它修改为上海时间 步骤 进入PostgreSQL的安装目录 /opt/1panel/apps/postgresql/postgresql/data打开postgresql.conf文件 修改&#xff1a; log_timezone Asia/Shanghai timezone Asia/Shanghai保存后重启…

uniapp vuex的使用

实现组件全局&#xff08;数据&#xff09;管理的一种机制&#xff0c;可以方便的实现组件之间共享数据&#xff0c;不同于上述三种传递值的方式。 可以把vuex当成一个store仓库&#xff0c;可以集中管理共享的数据&#xff0c;并且存储在vuex中的数据都是响应式的&#xff0c…

自动化新时代:机器取代工作,我们该如何重塑自我?

内容概要 在自动化时代的浪潮中&#xff0c;技术的飞速发展对传统工作模式产生了深远影响。我们眼前浮现的是一个充满机遇与挑战的新世界。许多岗位面临被机器取代的威胁&#xff0c;然而&#xff0c;这一变化并不仅仅是消极的。在这个背景下&#xff0c;个体不仅需要重新审视…

机器学习(基础2)

特征工程 特征工程:就是对特征进行相关的处理 一般使用pandas来进行数据清洗和数据处理、使用sklearn来进行特征工程 特征工程是将任意数据(如文本或图像)转换为可用于机器学习的数字特征,比如:字典特征提取(特征离散化)、文本特征提取、图像特征提取。 特征工程API 实例化…

Scala的Map集合

Map 有两种类型&#xff0c;可变与不可变&#xff0c;区别在于可变对象可以修改它&#xff0c;而不可变对象不可以。 默认情况下 Scala 使用不可变 Map。如果你需要使用可变集合&#xff0c;你需要显式的引入 import scala.collection.mutable.Map 类 在 Scala 中 你可以同时…