news 2026/6/15 17:10:52

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/6/15 14:32:04

IMU如何成为机器人自主移动的核心传感器

IMU是机器人感知自身运动状态的核心传感器&#xff0c;通过内置的加速度计与陀螺仪测量机器人的加速度与角速度&#xff0c;经算法解算后为机器人的控制、导航、平衡等功能提供关键数据&#xff0c;广泛应用于轮式机器人、足式机器人、无人机、机械臂等各类机器人系统中。核心功…

作者头像 李华
网站建设 2026/6/15 14:34:27

解析得物商品详情:API 接口调用实践与数据结构分析

得物平台&#xff08;原毒APP&#xff09;作为国内领先的潮流电商平台&#xff0c;其商品数据具有很高的价值。对于开发者而言&#xff0c;了解如何通过其提供的 API 接口获取商品详情信息&#xff0c;是进行数据分析、比价工具开发、库存监控等应用的基础。本文将聚焦于得物平…

作者头像 李华
网站建设 2026/6/15 14:32:04

车辆轨迹系统设计与实现

车辆轨迹系统设计与实现 一、设计背景与意义 车辆轨迹追踪在智能交通、物流车队管理、私家车防盗、网约车监管等领域具有重要应用价值。传统轨迹记录方式依赖车载终端与后台服务器简单交互&#xff0c;存在定位精度低、数据传输延迟、轨迹补全能力弱等问题&#xff0c;难以满足…

作者头像 李华
网站建设 2026/6/15 15:51:19

黑芝麻智能华山A2000 BaRT工具链:全场景智驾模型高效编译与部署

一、引言 随着智驾模型参数规模与计算复杂度指数级增长&#xff0c;大模型应用与车端硬件、性能功耗的矛盾愈发凸显&#xff0c;成为技术量产的关键瓶颈。在此背景下&#xff0c;黑芝麻智能华山A2000 BaRT 工具链以 MLIR 框架为基础&#xff0c;为辅助驾驶模型规模化、高性能、…

作者头像 李华
网站建设 2026/6/15 16:51:04

本年度必看!最佳信息登记二维码推荐榜单

在数字化的时代&#xff0c;信息登记二维码扮演着越来越重要的角色。通过二维码&#xff0c;企业、政府和个人可以高效管理信息&#xff0c;提升登记的便捷性和准确性。本文将从多个角度分析最佳信息登记二维码工具。每种工具都有其独特的功能和适用场景&#xff0c;让我们轻松…

作者头像 李华
网站建设 2026/6/15 14:52:29

django+Pythonuniapp的鲜花知识分享与销售商城系统的设计与实现小程序

文章目录 系统概述技术架构核心功能模块创新点应用价值 系统设计与实现的思路主要技术与实现手段源码lw获取/同行可拿货,招校园代理 &#xff1a;文章底部获取博主联系方式&#xff01; 系统概述 该系统基于Django后端框架与Python语言开发&#xff0c;结合UniApp跨平台前端技…

作者头像 李华