【后端面经】【缓存】36|Redis 单线程:为什么 Redis 用单线程而 Memcached 用多线程?

本文涉及的产品
Redis 开源版,标准版 2GB
推荐场景:
搭建游戏排行榜
简介: 【5月更文挑战第17天】Redis常被称为单线程,但实际上其在处理命令时采用单线程,但在6.0后IO变为多线程。持久化和数据同步等任务由额外线程处理,因此严格来说Redis是多线程的。面试时需理解Redis的IO模型,如epoll和Reactor模式,以及其内存操作带来的高性能。Redis使用epoll进行高效文件描述符管理,实现高性能的网络IO。在讨论Redis与Memcached的线程模型差异时,应强调Redis的单线程模型如何通过内存操作和高效IO实现高性能。

今天来探究下Redis高性能的原因。

Redis是单线程的含义

在学习Redis的时候肯定听说过一句话:Redis是单线程的。实际上,Redis并不是单线程的。业界说Redis是单线程的,是指它在处理命令的时候,是单线程的。在Redis6.0之前,Redis的IO也是单线程的,但是在6.0之后也改成了多线程。

但是其他部分,比如说持久化、数据同步之类的功能,都是由别的线程来完成的。因此严格来说,Redis其实是多线程的。

面试准备

这一部分的面试内容基本上是纯理论的,所以需要做几件事情。

  • 了解你使用的其他中间件,在IO上是否使用了epoll,以及是否使用了Reactor模式。

  • 了解公司有没有使用Redis的多线程,如果用了,那么弄清楚最开始的决策理由及相比单线程性能究竟提升了多少

  • 了解清楚你使用的Redis的性能瓶颈

如果你用的 Redis 真的启用了多线程模式,你就可以将这一点纳入到你的性能优化方案中。有关 Redis 的线程模型面试是纯理论面试,所以你需要记忆的东西很多。有时间的话可以把 Redis 的源码下载下来,看看和网络 IO 处理有关的部分,加深印象。

当你和面试官聊到了这些话题的时候,你就可以用这节课的知识来回答。

  • 网络 IO 问题。

  • 其他也用 epoll 的中间件。

  • 多线程的 Memcache,Memcache 用了多线程,但是 Redis 用了单线程。

  • Redis 的性能问题。

面试思路

一般来说,面试官都会问你“为什么Redis是单线程的,但是又能做到高性能?” 很多人都会下意识地回答:“因为Redis是完全内存操作的。” 这个理由很关键,但是并不是面试官想要的答案,他希望你回答得是Redis得IO模型。

所以要回答这个问题,首先要澄清Redis单线程得含义。

我们通常说的Redis单线程,其实是指处理命令得时候Redis是单线程得。但是Redis的其他部分,比如持久化其实是另外的线程在处理。因此本质上,Redis是多线程的。特别是Redis在6.0之后,连IO模型都改成了多线程的模型,进一步发挥了多核CPU的优势。

首先你先点明两个关键点:内存操作和高效IO模型

Redis的高性能源自两方面,一方面是Redis处理命令的时候,都是纯内存操作;另一方面,在Linux上Redis采用了epoll和Reactor结合的IO模型,非常高效。

这个时候他肯定就会问你,什么是 epoll,什么是 Reactor 模式。

epoll模型

简单来说就是epoll会帮你管着一大堆的套接字。每次你需要做啥的时候,就问问哪些套接字可用。读数据,就是找出那些已经收到了数据的套接字;写数据,就是找出那些可以写入数据的套接字。

而在Linux系统里,套接字就是一个普通的文件描述符,因此epoll本质上管着一堆文件描述符。
2024-05-19-20-48-36-image.png

epoll = CURD文件描述符

Redis使用的是epoll来处理IO。在Linux里面,万物都是文件,和网络IO有关的套接字也是文件。所以epoll要做的事情,就是管理这些文件描述符。或是用一句话来描述:epoll就是增删改查文件描述符。

目录
相关文章
|
4月前
|
安全 算法 Java
Java 多线程:线程安全与同步控制的深度解析
本文介绍了 Java 多线程开发的关键技术,涵盖线程的创建与启动、线程安全问题及其解决方案,包括 synchronized 关键字、原子类和线程间通信机制。通过示例代码讲解了多线程编程中的常见问题与优化方法,帮助开发者提升程序性能与稳定性。
182 0
|
4月前
|
数据采集 监控 调度
干货分享“用 多线程 爬取数据”:单线程 + 协程的效率反超 3 倍,这才是 Python 异步的正确打开方式
在 Python 爬虫中,多线程因 GIL 和切换开销效率低下,而协程通过用户态调度实现高并发,大幅提升爬取效率。本文详解协程原理、实战对比多线程性能,并提供最佳实践,助你掌握异步爬虫核心技术。
|
5月前
|
Java 数据挖掘 调度
Java 多线程创建零基础入门新手指南:从零开始全面学习多线程创建方法
本文从零基础角度出发,深入浅出地讲解Java多线程的创建方式。内容涵盖继承`Thread`类、实现`Runnable`接口、使用`Callable`和`Future`接口以及线程池的创建与管理等核心知识点。通过代码示例与应用场景分析,帮助读者理解每种方式的特点及适用场景,理论结合实践,轻松掌握Java多线程编程 essentials。
336 5
|
监控 NoSQL 安全
如何在 Redis 中正确使用多线程?
【10月更文挑战第16天】正确使用 Redis 多线程需要综合考虑多个因素,并且需要在实践中不断摸索和总结经验。通过合理的配置和运用,多线程可以为 Redis 带来性能上的提升,同时也要注意避免可能出现的问题,以保障系统的稳定和可靠运行。
324 2
|
8月前
|
缓存 NoSQL 中间件
Redis的线程模型
Redis采用单线程模型确保操作的原子性,每次只执行一个操作,避免并发冲突。它通过MULTI/EXEC事务机制、Lua脚本和复合指令(如MSET、GETSET等)保证多个操作要么全成功,要么全失败,确保数据一致性。Redis事务在EXEC前失败则不执行任何操作,EXEC后失败不影响其他操作。Pipeline虽高效但不具备原子性,适合非热点时段的数据调整。Redis 7引入Function功能,支持函数复用,简化复杂业务逻辑。总结来说,Redis的单线程模型简单高效,适用于高并发场景,但仍需合理选择指令执行方式以发挥其性能优势。
203 6
|
9月前
|
Python
python3多线程中使用线程睡眠
本文详细介绍了Python3多线程编程中使用线程睡眠的基本方法和应用场景。通过 `time.sleep()`函数,可以使线程暂停执行一段指定的时间,从而控制线程的执行节奏。通过实际示例演示了如何在多线程中使用线程睡眠来实现计数器和下载器功能。希望本文能帮助您更好地理解和应用Python多线程编程,提高程序的并发能力和执行效率。
317 20
|
9月前
|
安全 Java C#
Unity多线程使用(线程池)
在C#中使用线程池需引用`System.Threading`。创建单个线程时,务必在Unity程序停止前关闭线程(如使用`Thread.Abort()`),否则可能导致崩溃。示例代码展示了如何创建和管理线程,确保在线程中执行任务并在主线程中处理结果。完整代码包括线程池队列、主线程检查及线程安全的操作队列管理,确保多线程操作的稳定性和安全性。
|
11月前
|
NoSQL Redis
单线程传奇Redis,为何引入多线程?
Redis 4.0 引入多线程支持,主要用于后台对象删除、处理阻塞命令和网络 I/O 等操作,以提高并发性和性能。尽管如此,Redis 仍保留单线程执行模型处理客户端请求,确保高效性和简单性。多线程仅用于优化后台任务,如异步删除过期对象和分担读写操作,从而提升整体性能。
190 1
|
Java 开发者
在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口
【10月更文挑战第20天】在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口。本文揭示了这两种方式的微妙差异和潜在陷阱,帮助你更好地理解和选择适合项目需求的线程创建方式。
233 3
|
Java 开发者
在Java多线程编程中,选择合适的线程创建方法至关重要
【10月更文挑战第20天】在Java多线程编程中,选择合适的线程创建方法至关重要。本文通过案例分析,探讨了继承Thread类和实现Runnable接口两种方法的优缺点及适用场景,帮助开发者做出明智的选择。
126 2

热门文章

最新文章