news 2026/4/8 12:42:36

Redis 服务器线程与事件循环解析

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Redis 服务器线程与事件循环解析

文章目录

  • 事件循环线程
  • 事件循环相关源码文件
  • 事件主循环机制
  • 睡眠钩子(Hooks)机制
  • 睡眠钩子的实现
  • Epoll 实战:键盘输入监听

事件循环线程

众所周知,Redis 服务器启动时会开启六个线程:

  1. Thread-1 [redis-server](主线程)
  2. Thread-2 [bio_close_file](后台 I/O 线程 - 文件关闭)
  3. Thread-3 [bio_aof](后台 I/O 线程 - AOF 刷盘)
  4. Thread-4 [bio_lazy_free](后台 I/O 线程 - 惰性删除)
  5. Thread-5 [jemalloc_bg_thd](内存整理线程)
  6. Thread-6 [jemalloc_bg_thd](内存整理线程)

而事件循环则位于Thread-1 [redis-server](主线程)中。

事件循环相关源码文件

源码文件说明
ae.c事件驱动循环核心
ae_epoll.cLinux 系统实现(epoll)
ae_evport.cSolaris 专用
ae_kqueue.cmacOS 专用
ae_select.c老旧 Linux 或兼容模式使用
ebuckets.c事件桶管理
eventnotifier.c事件通讯器

事件主循环机制

Redis 的事件主循环位于ae.c中,核心代码如下:

voidaeMain(aeEventLoop*eventLoop){eventLoop->stop=0;while(!eventLoop->stop){aeProcessEvents(eventLoop,AE_ALL_EVENTS|AE_CALL_BEFORE_SLEEP|AE_CALL_AFTER_SLEEP);}}

但死循环非常消耗 CPU,Redis 是如何让线程进入睡眠等待的呢?

真正的“睡眠”发生在aeApiPoll中,其核心是调用系统epoll_wait

staticintaeApiPoll(aeEventLoop*eventLoop,structtimeval*tvp){aeApiState*state=eventLoop->apidata;intretval,numevents=0;retval=epoll_wait(state->epfd,state->events,eventLoop->setsize,tvp?(tvp->tv_sec*1000+(tvp->tv_usec+999)/1000):-1);if(retval>0){intj;numevents=retval;for(j=0;j<numevents;j++){intmask=0;structepoll_event*e=state->events+j;if(e->events&EPOLLIN)mask|=AE_READABLE;if(e->events&EPOLLOUT)mask|=AE_WRITABLE;if(e->events&EPOLLERR)mask|=AE_WRITABLE|AE_READABLE;if(e->events&EPOLLHUP)mask|=AE_WRITABLE|AE_READABLE;eventLoop->fired[j].fd=e->data.fd;eventLoop->fired[j].mask=mask;}}elseif(retval==-1&&errno!=EINTR){panic("aeApiPoll: epoll_wait, %s",strerror(errno));}returnnumevents;}
  • epoll_wait就是实现“睡眠”的系统调用。
  • 当没有事件时,线程在此阻塞,不消耗 CPU。

睡眠钩子(Hooks)机制

为了在睡眠前后执行必要逻辑,Redis 提供了钩子函数。

initServer函数中注册:

aeSetBeforeSleepProc(server.el, beforeSleep); aeSetAfterSleepProc(server.el, afterSleep);

aeProcessEvents中调用:

if (eventLoop->beforesleep != NULL && (flags & AE_CALL_BEFORE_SLEEP)) eventLoop->beforesleep(eventLoop); if (eventLoop->aftersleep != NULL && flags & AE_CALL_AFTER_SLEEP) eventLoop->aftersleep(eventLoop);

作用

  • beforeSleep:处理客户端写回、检查阻塞超时、计算下一次epoll_wait的超时时间等。
  • afterSleep:清理或重置休眠期间的状态。

这些钩子避免了无限循环中 CPU 空转,是 Redis 高效的核心之一。


睡眠钩子的实现

beforeSleep钩子和afterSleep钩子的实现都在server.c中,分别是void beforeSleep(struct aeEventLoop *eventLoop)和void afterSleep(struct aeEventLoop *eventLoop)。
实现上beforeSleep远比afterSleep复杂。

Epoll 实战:键盘输入监听

我们可以编写代码练习epoll的使用,模拟 Redis 的事件监听机制。

#include<stdio.h>#include<stdlib.h>#include<sys/epoll.h>#include<unistd.h>// 简单的错误处理宏#definepanic(fmt,...)\do{fprintf(stderr,"FATAL ERROR: "fmt" (%s:%d)\n",##__VA_ARGS__,__FILE__,__LINE__);\exit(1);}while(0)typedefstructepoll_eventEvent;intmain(void){intepfd=epoll_create(1);if(epfd<0){panic("epoll_create");return1;}// 设置监听标准输入(键盘)Event ev;ev.events=EPOLLIN;ev.data.fd=STDIN_FILENO;if(epoll_ctl(epfd,EPOLL_CTL_ADD,STDIN_FILENO,&ev)<0){panic("epoll_ctl");return1;}Event events[10];charbuf[1024];// 事件循环for(;;){intnfds=epoll_wait(epfd,events,10,-1);if(nfds<0){panic("epoll_wait");return1;}// 处理就绪事件for(inti=0;i<nfds;i++){if(events[i].data.fd==STDIN_FILENO){intnread=read(STDIN_FILENO,buf,sizeof(buf));if(nread>0){buf[nread]='\0';printf("键盘输入: %s",buf);}}}}close(epfd);return0;}

编译运行

gcc epoll_test.c -o epoll_test ./epoll_test

这个程序会在没有输入时“睡眠”,输入内容按回车后被唤醒,完美模拟了 Redis 的事件处理模型。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/8 8:22:40

超声波深度测量仪设计与实现

超声波深度测量仪设计与实现 一、设计背景与意义 超声波深度测量在水文监测、工业液位控制、水产养殖、污水处理等领域具有不可替代的作用。传统深度测量方法&#xff08;如浮标法、机械探测法&#xff09;存在操作复杂、测量滞后、易受环境干扰等问题&#xff0c;现有超声波…

作者头像 李华
网站建设 2026/4/7 22:54:02

【Redis持久化核心】AOF/RDB通俗详解+多场景对比

在讲核心内容前&#xff0c;先定一个前提&#xff1a;Redis是纯内存数据库&#xff0c;数据都存在内存里&#xff0c;一旦Redis重启/服务器断电&#xff0c;内存数据会直接消失。持久化就是Redis把内存数据“存到硬盘文件里”的操作&#xff0c;目的是重启后能恢复数据&#xf…

作者头像 李华
网站建设 2026/4/8 5:35:40

贡嘎山下的蓝色冰川,藏着海螺沟的旷世温柔

海螺沟位于四川省甘孜藏族自治州&#xff0c;其核心特点是同一区域内冰川、原始森林、温泉与雪山的罕见共存&#xff0c;形成了强烈而独特的视觉与地理反差。景区发源于贡嘎雪山东坡&#xff0c;主沟纵深超过30公里。现代海洋性冰川是这里的首要地质特征&#xff0c;巨大的冰舌…

作者头像 李华
网站建设 2026/4/8 1:50:30

django+Pythonuniapp的心理咨询信息系统APP小程序

文章目录技术栈与架构设计核心功能模块数据安全与合规性能优化策略部署与扩展性典型代码片段&#xff08;Django示例&#xff09;系统设计与实现的思路主要技术与实现手段源码lw获取/同行可拿货,招校园代理 &#xff1a;文章底部获取博主联系方式&#xff01;技术栈与架构设计 …

作者头像 李华
网站建设 2026/4/8 8:06:00

每天一个Linux命令_tar

tar 是 Tape Archive&#xff08;磁带归档&#xff09;的缩写 tar用来压缩和解压文件。tar本身不具有压缩功能。他是调用压缩功能实现的 打包&#xff08;不压缩&#xff09;&#xff1a;tar -cf 归档名.tar 源文件/目录 tar -cf testdir.tar testdir 解压&#xff08;不压…

作者头像 李华
网站建设 2026/4/8 5:35:58

生成式AI催生GEO优化,如何成为其内容权威信源?

生成式人工智能技术迅猛发展之际&#xff0c;一个叫GEO优化的全新概念于数字内容领域暗暗兴起。这种优化策略可不是传统搜索引擎优化即SEO的简单扩展&#xff0c;而是专门按照生成式AI的内容分发逻辑予以设定的一整套系统性办法。为理解GEO优化&#xff0c;就得从生成式AI的工作…

作者头像 李华