Redis源码解析1

前言 Redis(REmoteDIctionaryServer)是一个由Salvatore Sanfilippo写的key-value存储系统。 它有以下特点: 总体结构 Redis是一个单线程的服务器(除了写磁盘会开子进程,VM管理会开线程,先忽略这两块) 所以,它的服务器启动流程非常清晰,看下图,不再

前言

redis(remote dictionary server)是一个由salvatore sanfilippo写的key-value存储系统。
它有以下特点:

总体结构

Redis是一个单线程的服务器(除了写磁盘会开子进程,VM管理会开线程,先忽略这两块)
所以,它的服务器启动流程非常清晰,看下图,不再赘述

Redis源码解析1

 

 

事件循环1. 数据结构

通过 aeEventLoop 结构来表示一个事件框架,美国服务器,分析如下:

1 typedef struct aeEventLoop {timeEventNextId; aeFileEvent events[AE_SETSIZE]; aeFiredEvent fired[AE_SETSIZE]; aeTimeEvent *timeEventHead; stop;aeBeforeSleepProc *beforesleep;10 } aeEventLoop;

2. 事件函数分发流程

redis支持 epoll、kqueue、select 三种网络模型

网络框架的代码实现主要在以下几个文件中:

ae.c / ae.h      // 网络框架
ae_epoll.c       // epoll模型的实现
ae_kqueue.c   // kqueue模型的实现
ae_select.c     // select模型的实现

程序选择哪一种,网站空间,是在编译期确定的

1 1 file ae.c #ifdef HAVE_EPOLL#ifdef HAVE_KQUEUE#include

框架函数非常简单,从初始化到结束,主要的函数就3个
aeCreateEventLoop、aeMain、aeDeleteEventLoop
其中,aeMain是事件循环的主体函数,它又会调用 aeProcessEvents函数

三个主体函数会调用 aeApiCreate、aeApiPool、aeApiFree三个接口函数进行处理
这三个接口函数又会映射到具体的某一种网络模型中,而这是在编译期确定下来的

具体如下图所示:

代码小浣熊 代码小浣熊

代码小浣熊是基于商汤大语言模型的软件智能研发助手,覆盖软件需求分析、架构设计、代码编写、软件测试等环节

代码小浣熊 51 查看详情 代码小浣熊

Redis源码解析1

添加删除事件,由
aeCreateFileEvent、aeDeleteFileEvent函数完成,其函数分发流程如下图:

         

Redis源码解析1

2. 服务端socket建立流程

1. 在服务器初始化时,建立侦听socket,并绑定IP、Port
    支持 TCP连接 与本机的unixSocket连接

1if (server.port != 0) {2server.ipfd = anetTcpServer(server.neterr,server.port,server.bindaddr);3 … … 4 }5if (server.unixsocket != NULL) {server.sofd = anetUnixServer(server.neterr,server.unixsocket,server.unixsocketperm);8 … …9}

2. 侦听socket绑定到事件句柄

1if (server.ipfd > 0 && aeCreateFileEvent(server.el,server.ipfd,AE_READABLE,);3if (server.sofd > 0 && aeCreateFileEvent(server.el,server.sofd,AE_READABLE,);

其中“acceptTcpHandler”、“acceptUnixHandler” 是事件回调函数
当新连接到达时,会触发进入这两个函数

3. 再来看看 accept***Handler 里做了什么

1 void acceptTcpHandler(aeEventLoop *el, int fd, void *privdata, int mask) { 2int cport, cfd; 3char cip[128]; 4 REDIS_NOTUSED(el); 5 REDIS_NOTUSED(mask); 6 REDIS_NOTUSED(privdata);cfd = anetTcpAccept(server.neterr, fd, cip, &cport);10if (cfd == AE_ERR) {, server.neterr);12return;13 }, cip, cport);}

acceptCommonHandler中做的事很简单,调用 CreateClient函数,创建新的redisClient对象

acceptCommonHandler(int fd) {2redisClient *c;3if ((c = createClient(fd)) == NULL) {);;7 }8 … …9 }

在 createClient函数中,将新连接绑定到事件循环中

1 redisClient *createClient(int fd) { 2redisClient *c = zmalloc(sizeof(redisClient)); 3c->bufpos = 0; 4 5 anetNonBlock(NULL,fd); 6 anetTcpNoDelay(NULL,fd); 7if (aeCreateFileEvent(server.el,fd,AE_READABLE, {10 close(fd);11 zfree(c);12return NULL;13}

当连接上有数据到达时,便会触发 readQueryFromClient函数,进行实际的网络数据读取与处理

3. Timer事件

Redis将所有Timer绑定到事件循环中进行处理
通过函数 aeCreateTimeEvent 创建新的Timer事件
每一帧事件循环中,通过processTimeEvents函数进行处理

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/471504.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月8日 07:39:41
下一篇 2025年11月8日 07:44:38

相关推荐

  • php数据库如何实现数据缓存 php数据库减少查询压力的方案

    答案:PHP结合Redis等内存缓存系统可显著提升Web应用性能。通过将用户信息、热门数据等写入内存缓存并设置TTL,先查缓存未命中再查数据库,减少数据库压力;配合OPcache提升脚本执行效率,文件缓存适用于小型项目,数据库缓冲池优化和读写分离进一步提升性能,推荐Redis为主并防范缓存穿透与雪崩…

    2025年12月6日 后端开发
    000
  • win11提示“无法加载文件或程序集”错误怎么办_Win11文件或程序集无法加载修复方法

    首先运行SFC扫描修复系统文件,若无效则使用DISM修复系统映像,接着重装Visual C++运行库,再通过设置修复或重置出错应用,最后可手动替换并注册缺失的程序集文件。 如果您尝试在Windows 11系统中运行某个程序或打开文件时,收到“无法加载文件或程序集”的错误提示,则可能是由于系统文件损坏…

    2025年12月6日
    000
  • 如何在安装完成后优化缓存使用

    合理配置缓存策略可提升系统效率,需设置适宜的过期时间、选用多层存储介质并持续监控维护。 安装完成后优化缓存使用,关键在于合理配置缓存策略、选择合适的存储方式,并定期维护。以下是一些实用建议,帮助你提升系统或应用的缓存效率。 合理设置缓存过期时间 缓存的有效期直接影响数据的新鲜度和性能表现。设置过短会…

    2025年12月6日 数据库
    000
  • thinkphp队列任务不执行怎么解决

    答案是检查配置、推送、监听和执行环节。首先确认queue.php中驱动设置正确,如Redis或数据库配置无误;其次确保任务通过Queue::push正确推送到队列,并在对应存储查看记录;然后必须运行php think queue:work –daemon命令启动监听;最后检查fire方法…

    2025年12月6日 PHP框架
    000
  • PHP框架怎么使用缓存机制_PHP框架缓存驱动配置与数据缓存策略

    答案:现代PHP开发中,缓存通过减少数据库压力提升性能,主流框架如Laravel支持File、Redis、Memcached、Database等驱动,推荐生产环境使用Redis;合理设置过期时间、规范缓存键命名、条件性更新及分层缓存策略可优化性能,结合Cache::remember等方法实现高效数据…

    2025年12月6日 后端开发
    000
  • 使用PHP框架开发实时应用_基于Laravel的php框架怎么用的通信

    Laravel通过事件广播与WebSocket实现实时通信,1. 配置Redis或Pusher驱动并创建实现ShouldBroadcast接口的事件;2. 使用laravel/websockets扩展启动WebSocket服务器;3. 前端引入Laravel Echo连接本地WebSocket并监听…

    2025年12月6日 后端开发
    000
  • 如何在Laravel中实现缓存机制

    laravel的缓存机制用于提升应用性能,通过存储耗时操作结果避免重复计算。1. 配置缓存驱动:在.env文件中设置cache_driver,如redis,并安装相应扩展;2. 使用cache facade进行缓存操作,包括put、get、has、forget等方法;3. 使用remember和pu…

    2025年12月5日
    000
  • win8打开程序提示0xc000007b怎么办_win8程序0xc000007b错误解决方法

    首先重新安装Visual C++ Redistributable运行库,包括x86和x64版本;其次修复DirectX组件,更新至最新运行时;然后运行SFC扫描修复系统文件;最后手动注册vcruntime140.dll等关键DLL文件,每步完成后重启电脑测试程序。 如果您在Windows 8系统中尝…

    2025年12月5日
    000
  • Serverless架构下Workerman的无状态化改造方案

    在serverless架构下,workerman的无状态化改造可以通过以下步骤实现:1. 将workerman的逻辑拆分成独立的函数,如handleconnect、handlemessage和handleclose。2. 使用外部服务(如redis或dynamodb)存储状态信息。3. 采用事件驱动…

    2025年12月5日
    000
  • 应用程序发生异常0xc0000417怎么解决 5种解决方案揭晓

    方法一:重启系统并清理后台占用程序 有时候,出现“0xc0000417”错误只是因为系统临时资源冲突,尤其是在同时运行多个程序或后台任务过多的情况下。 1、先保存好当前的工作内容,然后重启计算机。 2、使用快捷键 Ctrl + Shift + Esc 调出任务管理器,查看正在运行的进程,关闭不必要的…

    2025年12月5日 电脑教程
    000
  • 电脑的dll文件丢失怎么恢复 一文教你快速恢复

    在使用计算机的过程中,不少用户可能会碰到类似的问题:启动某个程序或游戏时,系统弹出提示信息,例如“xxx.dll文件缺失”或“程序无法运行”。这类问题通常是由dll(动态链接库)文件被误删、损坏或丢失所引起的。本文将为你整理一套完整的应对策略,帮助你迅速找回丢失的dll文件,恢复正常系统运行。 一、…

    2025年12月5日 电脑教程
    000
  • 找不到Qt5Widgets.dll无法执行代码 解决方法推荐

    电脑在运行某些基于qt框架开发的应用程序时,可能会弹出提示:“无法找到qt5widgets.dll,代码执行无法继续。重新安装程序可能解决此问题。”这类错误通常由程序安装不完整、系统中关键dll文件丢失或环境变量配置异常引起,尤其在使用开源软件、跨平台工具或图形化界面应用时较为常见。本文将从三个实用…

    2025年12月5日 电脑教程
    000
  • 如何在Laravel中配置Redis缓存

    在laravel中配置redis缓存的核心步骤包括安装并运行redis服务、安装php扩展或composer包、配置.env文件和config/database.php、清除缓存。1. 安装redis服务器:使用系统包管理工具安装并启动redis服务;2. 安装php扩展或predis包:选择php…

    2025年12月5日
    000
  • 如何在Laravel中配置邮件发送服务

    laravel中配置邮件发送服务的核心是利用其邮件抽象层,通过修改.env文件和config/mail.php对接各种邮件服务商。1. 配置.env文件设置mail_mailer、mail_host、mail_port、mail_username、mail_password、mail_encrypt…

    2025年12月5日
    100
  • Java中如何实现限流 掌握流量控制

    在java中实现限流的方法主要包括计数器算法、滑动窗口算法、漏桶算法、令牌桶算法以及使用guava ratelimiter。1. 计数器算法通过设定时间窗口和请求数量进行限制,优点是实现简单,缺点是可能存在“突刺”问题;2. 滑动窗口算法将时间窗口细化,避免了“突刺”,效果更平滑但实现较复杂;3. …

    2025年12月5日 java
    000
  • PHP如何调用Scala代码 通过JVM桥接调用Scala程序的方法

    通过jvm桥接,php可调用scala代码,但需中间工具。具体步骤如下:1. 将scala代码编译为jar包,并确保类和方法为public;2. 部署javabridge到支持servlet的web服务器(如tomcat);3. 在php中配置java.inc并设置classpath以加载jar包;…

    2025年12月5日 后端开发
    000
  • 如何在Laravel中使用事件和监听器

    事件和监听器是laravel中实现松耦合的关键机制。1. 定义事件类如userregistered,封装发生的“事情”;2. 创建监听器如sendwelcomeemail,处理事件触发后的操作,并可异步执行;3. 在eventserviceprovider中注册事件与监听器的映射关系;4. 使用ev…

    2025年12月5日
    000
  • 如何在Laravel中配置队列工作器

    在laravel中配置队列工作器的核心步骤是设置队列驱动并启动监听进程,以提升应用性能和用户体验。1. 修改.env文件中的queue_connection变量,如设为redis以启用高性能队列;2. 配置redis连接信息确保其可用性;3. 使用php artisan queue:work命令启动…

    2025年12月5日
    100
  • 如何在Laravel中实现数据补全

    数据补全功能可在用户输入不完整信息时智能猜测并提供完整选项,laravel中可通过前后端协作实现:1.前端使用typeahead.js、select2或awesomplete监听输入框并发送ajax请求;2.后端创建路由和控制器接收输入值,用eloquent orm结合like或全文搜索技术查询数据…

    2025年12月5日
    000
  • 如何在Laravel中实现定时任务

    在laravel中实现定时任务,核心思路是利用框架的调度器集中管理任务,并通过服务器cron每分钟触发一次调度器执行。1. 创建命令:使用 php artisan make:command 生成命令类并编写业务逻辑;2. 注册任务:在 app/console/kernel.php 的 schedul…

    2025年12月5日
    000

发表回复

登录后才能评论
关注微信