[{"data":1,"prerenderedAt":3603},["ShallowReactive",2],{"search-docs":3,"doc-\u002Finterview\u002Fredis":886},[4,8,12,16,20,24,28,32,36,40,44,48,52,56,60,64,68,72,76,80,84,88,92,96,100,104,108,112,116,120,124,128,132,136,140,144,148,152,156,159,162,165,169,172,175,178,182,186,190,194,198,202,206,210,214,218,222,226,230,234,238,242,246,250,254,258,262,266,269,273,277,281,285,288,291,294,298,301,304,307,310,313,316,319,322,325,329,332,336,340,344,348,352,356,359,362,365,368,371,374,377,380,383,386,389,393,396,399,402,405,408,411,414,417,420,424,428,432,435,438,442,446,450,454,458,462,466,470,474,477,480,483,487,491,494,497,500,504,507,511,515,518,521,524,527,530,533,536,539,542,545,548,551,554,557,560,563,566,569,572,575,579,583,587,591,595,599,603,606,610,614,617,620,623,626,629,633,637,640,643,646,649,652,655,658,661,664,667,670,673,676,679,682,685,688,691,694,697,700,703,706,709,712,716,720,724,728,732,736,740,744,748,752,756,760,764,768,772,775,779,783,787,790,793,796,799,802,805,808,811,814,818,822,825,829,832,835,838,841,844,848,851,854,858,862,865,869,873,876,879,882],{"path":5,"title":6,"description":7},"\u002Fabout\u002Fauthor","作者相关","只想纯粹的做一个程序员...",{"path":9,"title":10,"description":11},"\u002Fabout\u002Fjourney","心路历程","",{"path":13,"title":14,"description":15},"\u002Fai\u002Fagent\u002Fframeworks","Agent 框架","主流 Agent 框架：LangChain、LlamaIndex、AutoGen、CrewAI",{"path":17,"title":18,"description":19},"\u002Fai\u002Fagent\u002Fhooks","Agent Hooks 与自动化","Claude Agent 的 Hooks 生命周期、事件类型、典型自动化场景",{"path":21,"title":22,"description":23},"\u002Fai\u002Fagent\u002Fintroduction","AI Agent 概述","AI Agent 核心概念：感知、规划、执行、记忆",{"path":25,"title":26,"description":27},"\u002Fai\u002Fagent\u002Fpractice","Agent 实战","AI Agent 实战：构建自主任务执行系统",{"path":29,"title":30,"description":31},"\u002Fai\u002Fagent\u002Fsdk","Claude Agent SDK 开发","使用 Claude Agent SDK 构建自定义 AI Agent：架构、API、生命周期",{"path":33,"title":34,"description":35},"\u002Fai\u002Fagent\u002Fsubagents","Subagents 子代理","用 Subagents 分解复杂任务、并发执行、隔离上下文",{"path":37,"title":38,"description":39},"\u002Fai\u002Fagent\u002Ftool-use","工具调用","AI Agent 工具调用：Function Calling、Tool Use 原理与实践",{"path":41,"title":42,"description":43},"\u002Fai\u002Ffundamentals\u002Fdeep-learning","深度学习入门","深度学习基础知识：前向传播、反向传播、损失函数、优化器",{"path":45,"title":46,"description":47},"\u002Fai\u002Ffundamentals\u002Fml-basics","机器学习基础","机器学习核心概念：监督学习、无监督学习、强化学习",{"path":49,"title":50,"description":51},"\u002Fai\u002Ffundamentals\u002Fneural-networks","神经网络原理","神经网络架构：CNN、RNN、注意力机制",{"path":53,"title":54,"description":55},"\u002Fai\u002Fgetting-started","AI 学习路线","AI 技术学习路线图，从基础到实战的完整指南",{"path":57,"title":58,"description":59},"\u002Fai\u002Fllm\u002Ffine-tuning","模型微调","大模型微调技术：LoRA、QLoRA、全量微调、RLHF",{"path":61,"title":62,"description":63},"\u002Fai\u002Fllm\u002Fintroduction","大模型概述","大语言模型发展历程、核心能力与主流模型对比",{"path":65,"title":66,"description":67},"\u002Fai\u002Fllm\u002Flocal-deploy","本地部署","大模型本地部署：Ollama、vLLM、llama.cpp",{"path":69,"title":70,"description":71},"\u002Fai\u002Fllm\u002Ftransformer","Transformer 架构","Transformer 架构详解：自注意力机制、位置编码、多头注意力",{"path":73,"title":74,"description":75},"\u002Fai\u002Fmcp\u002Fclient","MCP Client 开发","MCP Client 开发指南：连接、调用、集成",{"path":77,"title":78,"description":79},"\u002Fai\u002Fmcp\u002Fdebugging","MCP 调试与排错","MCP Server 开发与集成过程中的常见问题、日志分析、诊断工具",{"path":81,"title":82,"description":83},"\u002Fai\u002Fmcp\u002Fintroduction","MCP 概述","Model Context Protocol 协议概述：架构、核心概念、应用场景",{"path":85,"title":86,"description":87},"\u002Fai\u002Fmcp\u002Fserver","MCP Server 开发","MCP Server 开发指南：资源、工具、提示词的实现",{"path":89,"title":90,"description":91},"\u002Fai\u002Fmcp\u002Ftools","MCP Tools 深入","深入理解 MCP Tools：与 Resources\u002FPrompts 的差异、Schema 设计、Annotations 与权限控制",{"path":93,"title":94,"description":95},"\u002Fai\u002Fprompt\u002Fadvanced","高级 Prompt 模式","高级 Prompt 设计模式：Tree-of-Thought、自我反思、多轮对话策略",{"path":97,"title":98,"description":99},"\u002Fai\u002Fprompt\u002Fbasics","Prompt 基础","Prompt Engineering 入门：基本概念、角色设定、输出格式控制",{"path":101,"title":102,"description":103},"\u002Fai\u002Fprompt\u002Ftechniques","提示词技巧","常用提示词技巧：Few-shot、Chain-of-Thought、ReAct",{"path":105,"title":106,"description":107},"\u002Fai\u002Frag\u002Fembedding","文本嵌入","文本嵌入模型：Embedding 原理、模型选择、相似度计算",{"path":109,"title":110,"description":111},"\u002Fai\u002Frag\u002Fintroduction","RAG 概述","检索增强生成（RAG）架构原理、优势与应用场景",{"path":113,"title":114,"description":115},"\u002Fai\u002Frag\u002Fpractice","RAG 实战","RAG 应用实战：文档问答系统、知识库搭建",{"path":117,"title":118,"description":119},"\u002Fai\u002Frag\u002Fvector-database","向量数据库","主流向量数据库对比：Milvus、Pinecone、Chroma、Weaviate",{"path":121,"title":122,"description":123},"\u002Fai\u002Fskills\u002Fbest-practices","Skill 最佳实践","编写高质量 Skill 的设计原则、常见陷阱与优化技巧",{"path":125,"title":126,"description":127},"\u002Fai\u002Fskills\u002Fcreating","创建自定义 Skill","从零编写一个可被 Agent 自动发现和调用的 Skill",{"path":129,"title":130,"description":131},"\u002Fai\u002Fskills\u002Fintroduction","Agent Skills 概述","Claude Agent Skills 概念、工作原理、与 Tools\u002FMCP 的区别",{"path":133,"title":134,"description":135},"\u002Fgolang\u002Fadvanced\u002Fconcurrency","Go - 并发深入","深入理解 Go 并发编程的核心机制。",{"path":137,"title":138,"description":139},"\u002Fgolang\u002Fadvanced\u002Fgc","Go - 垃圾回收","理解 Go 的垃圾回收机制，掌握 GC 调优方法。",{"path":141,"title":142,"description":143},"\u002Fgolang\u002Fadvanced\u002Fgmp","Go - GMP 调度模型","GMP 是 Go 运行时调度器的核心模型，理解它对于编写高性能 Go 程序至关重要。",{"path":145,"title":146,"description":147},"\u002Fgolang\u002Fadvanced\u002Fgo-concurrency","Go - 并发编程","Go 的并发是其核心特性之一，通过 Goroutine 和 Channel 实现。",{"path":149,"title":150,"description":151},"\u002Fgolang\u002Fadvanced\u002Fmemory","Go - 内存模型","理解 Go 的内存分配机制和内存模型。",{"path":153,"title":154,"description":155},"\u002Fgolang\u002Fadvanced\u002Fprofiling","Go - 性能分析","掌握 Go 的性能分析工具：pprof、trace、benchmark。",{"path":157,"title":158,"description":11},"\u002Fgolang\u002Fcore\u002Fgo-basic","Go - 基础语法",{"path":160,"title":161,"description":11},"\u002Fgolang\u002Fcore\u002Fgo-composite","Go - 复合类型",{"path":163,"title":164,"description":11},"\u002Fgolang\u002Fcore\u002Fgo-control","Go - 流程控制",{"path":166,"title":167,"description":168},"\u002Fgolang\u002Fcore\u002Fgo-error","Go - 错误处理","Go 使用显式的错误返回值来处理错误，而不是异常机制。",{"path":170,"title":171,"description":11},"\u002Fgolang\u002Fcore\u002Fgo-function","Go - 函数",{"path":173,"title":174,"description":11},"\u002Fgolang\u002Fcore\u002Fgo-install","Go - 环境搭建",{"path":176,"title":177,"description":11},"\u002Fgolang\u002Fcore\u002Fgo-interface","Go - 接口",{"path":179,"title":180,"description":181},"\u002Fgolang\u002Fcore\u002Fgo-module","Go - 包管理","Go Modules 是 Go 1.11 引入的官方依赖管理方案，Go 1.16 后成为默认模式。",{"path":183,"title":184,"description":185},"\u002Fgolang\u002Fdistributed\u002Fgrpc","Go - gRPC","gRPC 是 Google 开发的高性能 RPC 框架，使用 Protocol Buffers 作为序列化协议。",{"path":187,"title":188,"description":189},"\u002Fgolang\u002Fdistributed\u002Fmicroservice","Go - 微服务","微服务架构的核心组件：服务发现、负载均衡、熔断降级。",{"path":191,"title":192,"description":193},"\u002Fgolang\u002Fdistributed\u002Fmq","Go - 消息队列","使用 Go 操作 Kafka 和 RabbitMQ。",{"path":195,"title":196,"description":197},"\u002Fgolang\u002Fdistributed\u002Fredis","Go - Redis","使用 go-redis 操作 Redis，实现缓存、分布式锁等功能。",{"path":199,"title":200,"description":201},"\u002Fgolang\u002Fengineering\u002Fconfig","Go - 配置管理","使用 viper 进行配置管理，支持多种配置格式和配置中心。",{"path":203,"title":204,"description":205},"\u002Fgolang\u002Fengineering\u002Fdocker","Go - Docker 部署","使用 Docker 容器化部署 Go 应用。",{"path":207,"title":208,"description":209},"\u002Fgolang\u002Fengineering\u002Fkubernetes","Go - Kubernetes 部署","在 Kubernetes 上部署和管理 Go 应用。",{"path":211,"title":212,"description":213},"\u002Fgolang\u002Fengineering\u002Flogging","Go - 日志系统","使用 zap 和 logrus 构建高性能结构化日志系统。",{"path":215,"title":216,"description":217},"\u002Fgolang\u002Fengineering\u002Ftesting","Go - 单元测试","Go 内置了强大的测试框架，掌握测试是编写高质量代码的基础。",{"path":219,"title":220,"description":221},"\u002Fgolang\u002Fstdlib\u002Fbufio","bufio","在 Go 语言中，bufio 包提供了带缓冲的 I\u002FO 操作，能够提高读写性能。以下是一些常用的 bufio 包 API 及其详细说明：",{"path":223,"title":224,"description":225},"\u002Fgolang\u002Fstdlib\u002Fcontainer","container","在Go语言标准库中，container 包提供了几种常用的数据结构实现，这些数据结构对于高效地管理和操作数据非常有用。以下是 container 包中主要的数据结构：",{"path":227,"title":228,"description":229},"\u002Fgolang\u002Fstdlib\u002Fcrypto","crypto","在 Go 语言中，crypto 包提供了一组用于加密和解密的功能。以下是一些常用的 crypto 包及其子包的 API 及其详细说明：",{"path":231,"title":232,"description":233},"\u002Fgolang\u002Fstdlib\u002Fencoding-csv","encoding\u002Fcsv","在 Go 语言中，encoding\u002Fcsv 包提供了对 CSV（逗号分隔值）文件进行读写的功能。以下是一些常用的 encoding\u002Fcsv 包的 API 及其详细说明：",{"path":235,"title":236,"description":237},"\u002Fgolang\u002Fstdlib\u002Fencoding-json","encoding\u002Fjson","在 Go 语言中，encoding\u002Fjson 包提供了对 JSON 数据进行编码和解码的功能。以下是一些常用的 encoding\u002Fjson 包的 API 及其详细说明：",{"path":239,"title":240,"description":241},"\u002Fgolang\u002Fstdlib\u002Fencoding-xml","encoding\u002Fxml","在 Go 语言中，encoding\u002Fxml 包提供了对 XML 数据进行编码和解码的功能。以下是一些常用的 encoding\u002Fxml 包的 API 及其详细说明：",{"path":243,"title":244,"description":245},"\u002Fgolang\u002Fstdlib\u002Fflag","flag","在Go语言中，flag 包是用于处理命令行参数的标准库，它提供了一种简单而直接的方式来解析和使用命令行参数。下面是关于 flag 包的一些基本介绍和常用功能：",{"path":247,"title":248,"description":249},"\u002Fgolang\u002Fstdlib\u002Ffmt","fmt","在 Go 语言的标准库中，fmt 包是非常重要的，它提供了处理格式化输入和输出的基本工具。以下是一些 fmt 包内常用的API：",{"path":251,"title":252,"description":253},"\u002Fgolang\u002Fstdlib\u002Fhttp","net\u002Fhttp","在 Go 语言中，net\u002Fhttp 包提供了用于构建 HTTP 客户端和服务器的强大工具。以下是一些常用的 net\u002Fhttp 包的 API 及其详细说明：",{"path":255,"title":256,"description":257},"\u002Fgolang\u002Fstdlib\u002Fio","io","在 Go 语言中，io 包提供了基本的输入输出功能。以下是一些常用的 io 包的 API 及其详细说明：",{"path":259,"title":260,"description":261},"\u002Fgolang\u002Fstdlib\u002Flog","log","在 Go 语言中，log 包提供了简单的日志记录功能。以下是一些常用的 log 包的 API 及其详细说明：",{"path":263,"title":264,"description":265},"\u002Fgolang\u002Fstdlib\u002Fmath","math","在 Go 语言中，math 包提供了基本的数学函数和常量。以下是一些常用的 math 包的 API 及其详细说明：",{"path":267,"title":268,"description":11},"\u002Fgolang\u002Fstdlib\u002Fnet","net",{"path":270,"title":271,"description":272},"\u002Fgolang\u002Fstdlib\u002Fos","os","在Go语言中，os 包是一个非常重要且常用的标准库，它提供了与操作系统交互的功能，包括文件操作、环境变量管理、进程管理等。下面是一些 os 包中常用的功能和API：",{"path":274,"title":275,"description":276},"\u002Fgolang\u002Fstdlib\u002Fsort","order","在 Go 语言中，sort 包提供了对切片和用户定义的集合进行排序的函数。它实现了常见的排序算法，如快速排序（Quicksort）和堆排序（Heapsort），并且为自定义集合提供了接口，使得用户可以根据特定的需求进行排序。",{"path":278,"title":279,"description":280},"\u002Fgolang\u002Fstdlib\u002Fstrconv","strconv","在 Go 语言中，strconv 包提供了字符串和基本数据类型之间的转换函数，例如将整数转换为字符串、字符串转换为整数，以及其他类型之间的转换。这些功能非常有用，特别是在处理用户输入或从外部数据源读取数据时。",{"path":282,"title":283,"description":284},"\u002Fgolang\u002Fstdlib\u002Ftime","time","在 Go 语言中，time 包提供了处理时间和日期的功能。以下是一些常用的 time 包的 API 及其详细说明：",{"path":286,"title":287,"description":11},"\u002Fgolang\u002Fweb\u002Fgin\u002Ferror","Gin - 错误处理",{"path":289,"title":290,"description":11},"\u002Fgolang\u002Fweb\u002Fgin\u002Ffile","Gin - 文件处理",{"path":292,"title":293,"description":11},"\u002Fgolang\u002Fweb\u002Fgin\u002Fmiddleware","Gin - 中间件",{"path":295,"title":296,"description":297},"\u002Fgolang\u002Fweb\u002Fgin\u002Fquickstart","Gin - 快速开始","Gin 是目前最流行的 Go Web 框架，以高性能和简洁 API 著称。",{"path":299,"title":300,"description":11},"\u002Fgolang\u002Fweb\u002Fgin\u002Frequest","Gin - 请求处理",{"path":302,"title":303,"description":11},"\u002Fgolang\u002Fweb\u002Fgin\u002Fresponse","Gin - 响应处理",{"path":305,"title":306,"description":11},"\u002Fgolang\u002Fweb\u002Fgin\u002Frouter","Gin - 路由",{"path":308,"title":309,"description":11},"\u002Fgolang\u002Fweb\u002Fgin\u002Fvalidation","Gin - 参数校验",{"path":311,"title":312,"description":11},"\u002Fgolang\u002Fweb\u002Fgorm\u002Fassociation","GORM - 关联关系",{"path":314,"title":315,"description":11},"\u002Fgolang\u002Fweb\u002Fgorm\u002Fcrud","GORM - CRUD 操作",{"path":317,"title":318,"description":11},"\u002Fgolang\u002Fweb\u002Fgorm\u002Fmodel","GORM - 模型定义",{"path":320,"title":321,"description":11},"\u002Fgolang\u002Fweb\u002Fgorm\u002Fperformance","GORM - 日志与性能",{"path":323,"title":324,"description":11},"\u002Fgolang\u002Fweb\u002Fgorm\u002Fquery","GORM - 高级查询",{"path":326,"title":327,"description":328},"\u002Fgolang\u002Fweb\u002Fgorm\u002Fquickstart","GORM - 快速开始","GORM 是 Go 语言最流行的 ORM 库，功能强大，使用简单。",{"path":330,"title":331,"description":11},"\u002Fgolang\u002Fweb\u002Fgorm\u002Ftransaction","GORM - 事务与 Hook",{"path":333,"title":334,"description":335},"\u002Finterview\u002Fbasic","计算机基础面经","本章节汇总了面试中常见的通用技术概念，不局限于特定语言或数据库，是考察技术内功的关键考点。",{"path":337,"title":338,"description":339},"\u002Finterview\u002Fgolang","Golang 面试题","Go 语言面试高频考点，覆盖基础语法、数据结构、并发编程、内存管理、GC、调度器等核心知识。",{"path":341,"title":342,"description":343},"\u002Finterview\u002Fk8s","Kubernetes 面试题","Kubernetes（K8s）面试高频考点，覆盖架构原理、核心资源、网络存储、调度策略、运维监控等核心知识。",{"path":345,"title":346,"description":347},"\u002Finterview\u002Fmysql","MySQL 面试题","MySQL 数据库面试高频考点，覆盖索引、事务、锁、优化、主从复制等核心知识。",{"path":349,"title":350,"description":351},"\u002Finterview\u002Fredis","Redis 面试题","Redis 面试高频考点，覆盖数据结构、持久化、集群、缓存一致性、性能优化等核心知识。",{"path":353,"title":354,"description":355},"\u002Finterview\u002Frocketmq","RocketMQ 面试题","RocketMQ 面试高频考点，覆盖消息模型、可靠性、顺序消息、事务消息、存储与高可用等核心知识。",{"path":357,"title":358,"description":11},"\u002Fother\u002Fjava\u002Fcollection\u002Flist-arraylist","List - ArrayList 源码解析",{"path":360,"title":361,"description":11},"\u002Fother\u002Fjava\u002Fcollection\u002Flist-linkedlist","List - LinkedList 源码解析",{"path":363,"title":364,"description":11},"\u002Fother\u002Fjava\u002Fcollection\u002Flist-stack","List - Satck源码解析",{"path":366,"title":367,"description":11},"\u002Fother\u002Fjava\u002Fcollection\u002Flist-vectore","List - Vector 源码解析",{"path":369,"title":370,"description":11},"\u002Fother\u002Fjava\u002Fcollection\u002Fmap-hashmap","Map - HashMap 源码解析",{"path":372,"title":373,"description":11},"\u002Fother\u002Fjava\u002Fcollection\u002Fmap-linkedhashmap","Map - LinkedHashMap 源码解析",{"path":375,"title":376,"description":11},"\u002Fother\u002Fjava\u002Fcollection\u002Fmap-treemap","Map - TreeMap 源码解析",{"path":378,"title":379,"description":11},"\u002Fother\u002Fjava\u002Fcollection\u002Fqueue-deque","Queue - Deque 接口解析",{"path":381,"title":382,"description":11},"\u002Fother\u002Fjava\u002Fcollection\u002Fqueue-queue","Queue - Queue 接口解析",{"path":384,"title":385,"description":11},"\u002Fother\u002Fjava\u002Fcollection\u002Fset-hashset","Set - HashSet源码解析",{"path":387,"title":388,"description":11},"\u002Fother\u002Fjava\u002Fcollection\u002Fset-linkedhashset","Set - LinkedHashSet 源码解析",{"path":390,"title":391,"description":392},"\u002Fother\u002Fjava\u002Fcollection\u002Fset-treeset","Set - TreeSet源码解析","TreeSet 是一个 Set 集合接口的实现类，与 HashSet 类似，其底层也是通过维护了一个 TreeMap 对象来封装了一些实现方法，故本篇不再对 TreeSet 的底层原理进行详细说明，仅对常用 API 做简单介绍，如需了解 TreeMap 的底层实现原理，请移步 Map - HashMap 源码解析",{"path":394,"title":395,"description":11},"\u002Fother\u002Fjava\u002Fcore\u002Fannotation","Java核心 - 注解",{"path":397,"title":398,"description":11},"\u002Fother\u002Fjava\u002Fcore\u002Fbasic-grammar","Java核心 - 基础语法",{"path":400,"title":401,"description":11},"\u002Fother\u002Fjava\u002Fcore\u002Fclass-and-object","Java核心 - 面向对象",{"path":403,"title":404,"description":11},"\u002Fother\u002Fjava\u002Fcore\u002Fcommon-classes","Java核心 - 常用类",{"path":406,"title":407,"description":11},"\u002Fother\u002Fjava\u002Fcore\u002Fexception","Java核心 - 异常处理",{"path":409,"title":410,"description":11},"\u002Fother\u002Fjava\u002Fcore\u002Fgenerics","Java核心 - 泛型",{"path":412,"title":413,"description":11},"\u002Fother\u002Fjava\u002Fcore\u002Fjdk-env-path","Java核心 - 环境搭建",{"path":415,"title":416,"description":11},"\u002Fother\u002Fjava\u002Fcore\u002Freflection","Java核心 - 反射",{"path":418,"title":419,"description":11},"\u002Fother\u002Fjava\u002Fcore\u002Fstring","Java核心 - String 字符串",{"path":421,"title":422,"description":423},"\u002Fother\u002Fjava\u002Fio\u002Fbuffer-stream","Java IO - 缓冲流","缓冲流是对基本流的包装，通过内置缓冲区减少系统调用次数，大幅提升读写效率。",{"path":425,"title":426,"description":427},"\u002Fother\u002Fjava\u002Fio\u002Fbyte-stream","Java IO - 字节流","字节流是 Java IO 中最基本的流类型，以字节（byte）为单位进行数据读写，可以处理任意类型的文件。",{"path":429,"title":430,"description":431},"\u002Fother\u002Fjava\u002Fio\u002Fchar-stream","Java IO - 字符流","字符流以字符为单位进行读写，专门用于处理文本文件。相比字节流，字符流能够正确处理字符编码，避免中文乱码问题。",{"path":433,"title":434,"description":11},"\u002Fother\u002Fjava\u002Fio\u002Ffile","Java IO - File 类",{"path":436,"title":437,"description":11},"\u002Fother\u002Fjava\u002Fio\u002Fio-stream-system","Java IO - IO流概述",{"path":439,"title":440,"description":441},"\u002Fother\u002Fjava\u002Fio\u002Fnio","Java IO - NIO","NIO（New IO）是 JDK 1.4 引入的新 IO 模型，提供了更高效的 IO 操作方式，支持非阻塞 IO 和多路复用。",{"path":443,"title":444,"description":445},"\u002Fother\u002Fjava\u002Fjvm\u002Fclass-loading","类加载机制","类加载机制是 JVM 将 .class 文件加载到内存，并对数据进行校验、转换解析和初始化，最终形成可被 JVM 直接使用的 Java 类型的过程。",{"path":447,"title":448,"description":449},"\u002Fother\u002Fjava\u002Fjvm\u002Fgarbage-collection","垃圾回收","垃圾回收（Garbage Collection，GC）是 JVM 自动管理内存的机制，负责回收不再使用的对象所占用的内存。",{"path":451,"title":452,"description":453},"\u002Fother\u002Fjava\u002Fjvm\u002Fjvm-memory","JVM 内存结构","JVM 在执行 Java 程序时，会把它管理的内存划分为若干个不同的数据区域。这些区域有各自的用途、创建和销毁时间。",{"path":455,"title":456,"description":457},"\u002Fother\u002Fjava\u002Fjvm\u002Fjvm-tuning","JVM 调优","JVM 调优是优化 Java 应用性能的重要手段，主要包括参数配置、性能监控和问题排查。",{"path":459,"title":460,"description":461},"\u002Fother\u002Fjava\u002Fthread\u002Fatomic","原子类","Java 原子类（Atomic Classes）提供了一种无锁的线程安全方式，基于 CAS（Compare-And-Swap）操作实现。",{"path":463,"title":464,"description":465},"\u002Fother\u002Fjava\u002Fthread\u002Fcompletable-future","CompletableFuture","CompletableFuture 是 JDK 8 引入的异步编程工具，实现了 Future 和 CompletionStage 接口，支持函数式编程和链式调用。",{"path":467,"title":468,"description":469},"\u002Fother\u002Fjava\u002Fthread\u002Fconcurrent-collections","并发集合","Java 并发包提供了多种线程安全的集合类，用于替代传统的同步集合（如 Collections.synchronizedList）。",{"path":471,"title":472,"description":473},"\u002Fother\u002Fjava\u002Fthread\u002Fconcurrent-utils","并发工具类","Java 并发包提供了多种实用的并发工具类，用于控制线程之间的协调与同步。",{"path":475,"title":476,"description":11},"\u002Fother\u002Fjava\u002Fthread\u002Fsynchronized-lock","同步机制",{"path":478,"title":479,"description":11},"\u002Fother\u002Fjava\u002Fthread\u002Fthread-basic","线程基础",{"path":481,"title":482,"description":11},"\u002Fother\u002Fjava\u002Fthread\u002Fthread-pool","线程池",{"path":484,"title":485,"description":486},"\u002Fother\u002Fspring-series\u002Fspring\u002Fannotations-beans","Spring - 基于注解管理Bean","从 Java 5 开始，Java 增加了对注解（Annotation）的支持，它是代码中的一种特殊标记，可以在编译、类加载和运行时被读取，执行相应的处理。开发人员可以通过注解在不改变原有代码和逻辑的情况下，在源代码中嵌入补充信息。",{"path":488,"title":489,"description":490},"\u002Fother\u002Fspring-series\u002Fspring\u002Fimplement-ioc","Spring - 原理手写IoC","Spring 框架的 IOC 是基于 Java 反射机制实现的，在学习手写 IoC 之前，你需要具备一定的 Java 反射相关的知识，参考本站内的 Java 教程。",{"path":492,"title":493,"description":11},"\u002Fother\u002Fspring-series\u002Fspring\u002Fintroduction-case","Spring - 入门案例",{"path":495,"title":496,"description":11},"\u002Fother\u002Fspring-series\u002Fspring\u002Fspring-aop","Spring - 面向切面AOP",{"path":498,"title":499,"description":11},"\u002Fother\u002Fspring-series\u002Fspring\u002Fspring-aot","Spring - AOT提前编译",{"path":501,"title":502,"description":503},"\u002Fother\u002Fspring-series\u002Fspring\u002Fspring-data-validation","Spring - 数据校验","在开发中，我们经常遇到参数校验的需求，比如用户注册的时候，要校验用户名不能为空、用户名长度不超过20个字符、手机号是合法的手机号格式等等。如果使用普通方式，我们会把校验的代码和真正的业务处理逻辑耦合在一起，而且如果未来要新增一种校验逻辑也需要在修改多个地方。而spring validation允许通过注解的方式来定义对象校验规则，把校验和业务逻辑分离开，让代码编写更加方便。Spring Validation其实就是对Hibernate Validator进一步的封装，方便在Spring中使用。",{"path":505,"title":506,"description":11},"\u002Fother\u002Fspring-series\u002Fspring\u002Fspring-i18n","Spring - 国际化i18n",{"path":508,"title":509,"description":510},"\u002Fother\u002Fspring-series\u002Fspring\u002Fspring-ioc","Spring - IOC容器","IoC 是 Inversion of Control 的简写，译为“控制反转”，它不是一门技术，而是一种设计思想，是一个重要的面向对象编程法则，能够指导我们如何设计出松耦合、更优良的程序。",{"path":512,"title":513,"description":514},"\u002Fother\u002Fspring-series\u002Fspring\u002Fspring-junit","Spring - 单元测试JUnit","在之前的测试方法中，几乎都能看到以下的两行代码：",{"path":516,"title":517,"description":11},"\u002Fother\u002Fspring-series\u002Fspring\u002Fspring-resources","Spring - 资源操作",{"path":519,"title":520,"description":11},"\u002Fother\u002Fspring-series\u002Fspring\u002Fspring-summarize","Spring - Spring概述",{"path":522,"title":523,"description":11},"\u002Fother\u002Fspring-series\u002Fspring\u002Fspring-transaction","Spring - 事务",{"path":525,"title":526,"description":11},"\u002Fother\u002Fspring-series\u002Fspring\u002Fxml-beans","Spring - 基于XML管理Bean",{"path":528,"title":529,"description":11},"\u002Fother\u002Fspring-series\u002Fspringboot\u002Fspringboot-config","SpringBoot - 配置详解",{"path":531,"title":532,"description":11},"\u002Fother\u002Fspring-series\u002Fspringboot\u002Fspringboot-data","SpringBoot - 数据访问",{"path":534,"title":535,"description":11},"\u002Fother\u002Fspring-series\u002Fspringboot\u002Fspringboot-quickstart","SpringBoot - 快速入门",{"path":537,"title":538,"description":11},"\u002Fother\u002Fspring-series\u002Fspringboot\u002Fspringboot-web","SpringBoot - Web 开发",{"path":540,"title":541,"description":11},"\u002Fother\u002Fspring-series\u002Fspringcloud\u002Fspringcloud-config","SpringCloud - 配置中心",{"path":543,"title":544,"description":11},"\u002Fother\u002Fspring-series\u002Fspringcloud\u002Fspringcloud-discovery","SpringCloud - 服务注册与发现",{"path":546,"title":547,"description":11},"\u002Fother\u002Fspring-series\u002Fspringcloud\u002Fspringcloud-feign","SpringCloud - 服务调用",{"path":549,"title":550,"description":11},"\u002Fother\u002Fspring-series\u002Fspringcloud\u002Fspringcloud-gateway","SpringCloud - 服务网关",{"path":552,"title":553,"description":11},"\u002Fother\u002Fspring-series\u002Fspringcloud\u002Fspringcloud-introduction","SpringCloud - 微服务概述",{"path":555,"title":556,"description":11},"\u002Fother\u002Fspring-series\u002Fspringcloud\u002Fspringcloud-sentinel","SpringCloud - 服务保护",{"path":558,"title":559,"description":11},"\u002Fother\u002Fspring-series\u002Fspringmvc\u002Fspringmvc-databind","SpringMVC - 数据绑定与转换",{"path":561,"title":562,"description":11},"\u002Fother\u002Fspring-series\u002Fspringmvc\u002Fspringmvc-exception","SpringMVC - 异常处理",{"path":564,"title":565,"description":11},"\u002Fother\u002Fspring-series\u002Fspringmvc\u002Fspringmvc-interceptor","SpringMVC - 拦截器",{"path":567,"title":568,"description":11},"\u002Fother\u002Fspring-series\u002Fspringmvc\u002Fspringmvc-introduction","SpringMVC - 简介与环境搭建",{"path":570,"title":571,"description":11},"\u002Fother\u002Fspring-series\u002Fspringmvc\u002Fspringmvc-request","SpringMVC - 请求处理",{"path":573,"title":574,"description":11},"\u002Fother\u002Fspring-series\u002Fspringmvc\u002Fspringmvc-response","SpringMVC - 响应处理",{"path":576,"title":577,"description":578},"\u002Fproject\u002Frocket-leaf\u002Farchitecture","项目架构","Rocket-Leaf 的目录结构、模块划分、数据流向，以及各层之间的依赖关系。",{"path":580,"title":581,"description":582},"\u002Fproject\u002Frocket-leaf\u002Fbackend-layers","后端分层设计","Rocket-Leaf 的 model \u002F rocketmq \u002F service 三层结构，以及服务之间的依赖关系与设计取舍。",{"path":584,"title":585,"description":586},"\u002Fproject\u002Frocket-leaf\u002Fclient-manager","RocketMQ 客户端管理器","AdminClientManager 的多客户端池、默认连接懒加载、自动重连重试的设计与实现。",{"path":588,"title":589,"description":590},"\u002Fproject\u002Frocket-leaf\u002Fencryption","连接信息加密存储","AES-256-GCM + SHA-256 字段级派生密钥的实现，以及如何在不破坏兼容性的前提下为历史明文数据做透明迁移。",{"path":592,"title":593,"description":594},"\u002Fproject\u002Frocket-leaf\u002Ffrontend","前端结构与类型绑定","React + Vite 目录组织、自动生成的 Wails 绑定、api 薄封装与自定义 hooks 的职责划分。",{"path":596,"title":597,"description":598},"\u002Fproject\u002Frocket-leaf","项目简介","Rocket-Leaf 是一款基于 Wails v3 构建的跨平台 RocketMQ 桌面管理客户端，Go 后端 + React 前端。本文档系列拆解它的架构与关键实现。",{"path":600,"title":601,"description":602},"\u002Fproject\u002Frocket-leaf\u002Fwails-v3","Wails v3 入门","Wails v3 的核心概念、Service 绑定机制，以及 Rocket-Leaf 是如何用它把 Go 后端和 React 前端打通的。",{"path":604,"title":605,"description":11},"\u002Ftutorials\u002Fcloud\u002Fdocker\u002Fdocker-basic","Docker - 入门基础",{"path":607,"title":608,"description":609},"\u002Ftutorials\u002Fcloud\u002Fdocker\u002Fdocker-compose","Docker - Compose","在部署应用时，常常使用到不止一个容器，那么在部署容器的时候就需要一个一个进行部署，这样的部署过程也相对来说比较繁琐复杂，也容易出问题，那么有没有一种更为简单的方法呢？",{"path":611,"title":612,"description":613},"\u002Ftutorials\u002Fcloud\u002Fdocker\u002Fdocker-container-connection","Docker - 容器互联","在上一个章节中我们学习了 Docker 容器的端口映射，可以将 Docker 容器和本地以及网络中的端口进行连接起来。",{"path":615,"title":616,"description":11},"\u002Ftutorials\u002Fcloud\u002Fdocker\u002Fdocker-dockerfile","Docker - Dockerfile",{"path":618,"title":619,"description":11},"\u002Ftutorials\u002Fcloud\u002Fdocker\u002Fdocker-helloworld","Docker - HelloWorld",{"path":621,"title":622,"description":11},"\u002Ftutorials\u002Fcloud\u002Fdocker\u002Fdocker-install","Docker - 安装",{"path":624,"title":625,"description":11},"\u002Ftutorials\u002Fcloud\u002Fdocker\u002Fdocker-introduce","Docker - 简介",{"path":627,"title":628,"description":11},"\u002Ftutorials\u002Fcloud\u002Fdocker\u002Fdocker-object","Docker - 镜像、容器、仓库",{"path":630,"title":631,"description":632},"\u002Ftutorials\u002Fcloud\u002Fdocker\u002Fdocker-warehouse","Docker - 仓库管理","仓库是集中存放资源的地方，代码仓库是存放代码的，那么Docker 中的仓库就是存放 Docker 镜像的。",{"path":634,"title":635,"description":636},"\u002Ftutorials\u002Fcloud\u002Fdocker\u002Fdocker-web-containers","Docker - WEB应用实例","在之前的章节中，仅对普通容器进行了演示，但在实际中常常使用到 Docker 容器中的 WEB 应用程序。",{"path":638,"title":639,"description":11},"\u002Ftutorials\u002Fcloud\u002Fkubernetes\u002Fk8s-config","Kubernetes - ConfigMap 与 Secret",{"path":641,"title":642,"description":11},"\u002Ftutorials\u002Fcloud\u002Fkubernetes\u002Fk8s-helm","Kubernetes - Helm 包管理",{"path":644,"title":645,"description":11},"\u002Ftutorials\u002Fcloud\u002Fkubernetes\u002Fk8s-install","Kubernetes - 集群安装",{"path":647,"title":648,"description":11},"\u002Ftutorials\u002Fcloud\u002Fkubernetes\u002Fk8s-introduction","Kubernetes - 简介与架构",{"path":650,"title":651,"description":11},"\u002Ftutorials\u002Fcloud\u002Fkubernetes\u002Fk8s-kubectl","Kubernetes - kubectl 命令行工具",{"path":653,"title":654,"description":11},"\u002Ftutorials\u002Fcloud\u002Fkubernetes\u002Fk8s-monitoring","Kubernetes - 监控与日志",{"path":656,"title":657,"description":11},"\u002Ftutorials\u002Fcloud\u002Fkubernetes\u002Fk8s-network-security","Kubernetes - 网络与安全",{"path":659,"title":660,"description":11},"\u002Ftutorials\u002Fcloud\u002Fkubernetes\u002Fk8s-service","Kubernetes - Service 与 Ingress",{"path":662,"title":663,"description":11},"\u002Ftutorials\u002Fcloud\u002Fkubernetes\u002Fk8s-storage","Kubernetes - 持久化存储",{"path":665,"title":666,"description":11},"\u002Ftutorials\u002Fcloud\u002Fkubernetes\u002Fk8s-workload","Kubernetes - 工作负载资源",{"path":668,"title":669,"description":11},"\u002Ftutorials\u002Fcloud\u002Flinux\u002Flinux-bash","Linux - Bash 基础语法",{"path":671,"title":672,"description":11},"\u002Ftutorials\u002Fcloud\u002Flinux\u002Flinux-file-directory","Linux - 文件与目录操作",{"path":674,"title":675,"description":11},"\u002Ftutorials\u002Fcloud\u002Flinux\u002Flinux-network","Linux - 网络配置",{"path":677,"title":678,"description":11},"\u002Ftutorials\u002Fcloud\u002Flinux\u002Flinux-package","Linux - 软件包管理",{"path":680,"title":681,"description":11},"\u002Ftutorials\u002Fcloud\u002Flinux\u002Flinux-process","Linux - 进程管理",{"path":683,"title":684,"description":11},"\u002Ftutorials\u002Fcloud\u002Flinux\u002Flinux-scripts","Linux - 常用脚本示例",{"path":686,"title":687,"description":11},"\u002Ftutorials\u002Fcloud\u002Flinux\u002Flinux-service","Linux - 服务管理",{"path":689,"title":690,"description":11},"\u002Ftutorials\u002Fcloud\u002Flinux\u002Flinux-user-permission","Linux - 用户与权限管理",{"path":692,"title":693,"description":11},"\u002Ftutorials\u002Fcloud\u002Fnginx\u002Fnginx-https","Nginx - HTTPS 配置",{"path":695,"title":696,"description":11},"\u002Ftutorials\u002Fcloud\u002Fnginx\u002Fnginx-install","Nginx - 安装与配置",{"path":698,"title":699,"description":11},"\u002Ftutorials\u002Fcloud\u002Fnginx\u002Fnginx-loadbalance","Nginx - 负载均衡",{"path":701,"title":702,"description":11},"\u002Ftutorials\u002Fcloud\u002Fnginx\u002Fnginx-optimization","Nginx - 性能优化",{"path":704,"title":705,"description":11},"\u002Ftutorials\u002Fcloud\u002Fnginx\u002Fnginx-proxy","Nginx - 反向代理",{"path":707,"title":708,"description":11},"\u002Ftutorials\u002Fcloud\u002Fnginx\u002Fnginx-static","Nginx - 静态资源服务",{"path":710,"title":711,"description":11},"\u002Ftutorials\u002Fcloud\u002Fnginx\u002Fnginx-vhost","Nginx - 虚拟主机配置",{"path":713,"title":714,"description":715},"\u002Ftutorials\u002Fdatabase\u002Fmysql\u002Fmysql-architecture","MySQL 高可用架构","主从复制、读写分离、分库分表。",{"path":717,"title":718,"description":719},"\u002Ftutorials\u002Fdatabase\u002Fmysql\u002Fmysql-index","MySQL 索引","索引是帮助 MySQL 高效获取数据的有序数据结构。",{"path":721,"title":722,"description":723},"\u002Ftutorials\u002Fdatabase\u002Fmysql\u002Fmysql-lock","MySQL 锁","锁用于解决并发访问时的数据一致性问题。",{"path":725,"title":726,"description":727},"\u002Ftutorials\u002Fdatabase\u002Fmysql\u002Fmysql-optimize","MySQL 性能优化","SQL 优化是后端开发必备技能。",{"path":729,"title":730,"description":731},"\u002Ftutorials\u002Fdatabase\u002Fmysql\u002Fmysql-transaction","MySQL 事务","事务是一组不可分割的操作，要么全部成功，要么全部失败。",{"path":733,"title":734,"description":735},"\u002Ftutorials\u002Fdatabase\u002Fmysql\u002Fsql-advanced","SQL 进阶","多表查询、子查询、函数、视图、存储过程。",{"path":737,"title":738,"description":739},"\u002Ftutorials\u002Fdatabase\u002Fmysql\u002Fsql-basic","SQL 基础","SQL（Structured Query Language）是操作关系型数据库的标准语言。",{"path":741,"title":742,"description":743},"\u002Ftutorials\u002Fdatabase\u002Fredis\u002Fredis-advanced","Redis 进阶功能","事务、发布订阅、Lua 脚本、Pipeline。",{"path":745,"title":746,"description":747},"\u002Ftutorials\u002Fdatabase\u002Fredis\u002Fredis-basic","Redis 基础","Redis 安装配置与基本命令。",{"path":749,"title":750,"description":751},"\u002Ftutorials\u002Fdatabase\u002Fredis\u002Fredis-cluster","Redis 高可用","主从复制、哨兵、Cluster 集群。",{"path":753,"title":754,"description":755},"\u002Ftutorials\u002Fdatabase\u002Fredis\u002Fredis-datatype","Redis 数据类型","Redis 5 种基本数据类型 + 4 种特殊类型。",{"path":757,"title":758,"description":759},"\u002Ftutorials\u002Fdatabase\u002Fredis\u002Fredis-optimize","Redis 性能优化","内存优化、缓存问题、最佳实践。",{"path":761,"title":762,"description":763},"\u002Ftutorials\u002Fdatabase\u002Fredis\u002Fredis-persistence","Redis 持久化","Redis 提供 RDB 和 AOF 两种持久化方式。",{"path":765,"title":766,"description":767},"\u002Ftutorials\u002Fdatabase\u002Fredis\u002Fredis-principle","Redis 底层原理","数据结构、线程模型、网络模型。",{"path":769,"title":770,"description":771},"\u002Ftutorials\u002Fdev-idea\u002Fdesign-patterns\u002Fbehaiver-patterns\u002Fobserver-pattern","观察者模式","观察者模式属于行为型模式，定义了对象之间的一对多的依赖关系，在这种模式中，当一个对象的状态发生变化时，所有依赖于它的对象都会得到通知，并且执行相关操作。观察者模式又被成为“发布—订阅模式”，即发布者发生改变后，会通知所有订阅者。",{"path":773,"title":774,"description":11},"\u002Ftutorials\u002Fdev-idea\u002Fdesign-patterns\u002Fcreate-patterns\u002Ffactory-pattern","工厂模式",{"path":776,"title":777,"description":778},"\u002Ftutorials\u002Fdev-idea\u002Fdesign-patterns\u002Fcreate-patterns\u002Fsingleton-pattern","单例模式","单例模式是最常用的设计模式之一，他可以保证在整个应用中，某个类只存在一个实例化对象，即全局使用到该类的只有一个对象，这种模式在需要限制某些类的实例数量时非常有用，通常全局只需要一个该对象即可，如一些配置文件映射对象、数据库连接对象等。",{"path":780,"title":781,"description":782},"\u002Ftutorials\u002Fdev-idea\u002Fdesign-patterns\u002Fstructural-patterns\u002Fadapter-pattern","适配器模式","适配器模式是一种结构型模式，可以将一个类的接口转换成客户端所期望的另一种接口，适配器模式可以帮助开发人员在不修改现有代码的情况下，将不兼容的类组合在一起。",{"path":784,"title":785,"description":786},"\u002Ftutorials\u002Fdev-tools\u002Fgit\u002Fgit-basic-operations","Git 创建版本库","在 Git 上创建版本库有两种方式，一种是直接拷贝远程 Git 仓库到本地，另外一种是我们自己创建本地的版本库。",{"path":788,"title":789,"description":11},"\u002Ftutorials\u002Fdev-tools\u002Fgit\u002Fgit-branch-manage","Git 分支管理",{"path":791,"title":792,"description":11},"\u002Ftutorials\u002Fdev-tools\u002Fgit\u002Fgit-content-operations","Git 仓库内容操作",{"path":794,"title":795,"description":11},"\u002Ftutorials\u002Fdev-tools\u002Fgit\u002Fgit-introduce-install","Git 介绍和安装",{"path":797,"title":798,"description":11},"\u002Ftutorials\u002Fdev-tools\u002Fgit\u002Fgit-remote-manage","Git 远程管理",{"path":800,"title":801,"description":11},"\u002Ftutorials\u002Fdev-tools\u002Fgit\u002Fgit-workspace-index-repo","Git 工作原理",{"path":803,"title":804,"description":11},"\u002Ftutorials\u002Fdev-tools\u002Fhomebrew","HomeBrew 教程",{"path":806,"title":807,"description":11},"\u002Ftutorials\u002Fdev-tools\u002Fidea\u002Fshortcuts","快捷键",{"path":809,"title":810,"description":11},"\u002Ftutorials\u002Fdev-tools\u002Fmaven\u002Fintroduce-install-config","Maven - 介绍、安装、配置",{"path":812,"title":813,"description":11},"\u002Ftutorials\u002Ffront-end\u002Fvue3\u002Fbasic-knowledge","2. 基础知识",{"path":815,"title":816,"description":817},"\u002Ftutorials\u002Ffront-end\u002Fvue3\u002Fcomponent-communication","9. 组件通信","在前面的章节内，介绍了 Vue 中最核心的内容——组件的介绍和使用，和 Java 等编程语言相反，组件并不近似于这些变成语言中的类，类可以通过类或者其实例化的对象来相互交互，但 Vue 组件之间的作用域是相互独立的，这就意味着不同组件之间的数据无法相互引用。",{"path":819,"title":820,"description":821},"\u002Ftutorials\u002Ffront-end\u002Fvue3\u002Fcomputed","4. 计算属性","虽然直接在模板中使用表达式方便，但是如果在模板中添加很多逻辑，会让模板变的臃肿且难维护，耦合度较高。有没有一种简单的方式来实现呢？答案是有的。",{"path":823,"title":824,"description":11},"\u002Ftutorials\u002Ffront-end\u002Fvue3\u002Fcreate-vue-project","1. 环境搭建及安装",{"path":826,"title":827,"description":828},"\u002Ftutorials\u002Ffront-end\u002Fvue3\u002Flife-cycle","6. 生命周期","生命周期是指组件从创建、挂载、更新到销毁的整个过程中所经历的一系列阶段。在 Vue 中，每个组件都有自己的生命周期，可以通过生命周期钩子函数来监听和处理组件在不同阶段的行为和状态。",{"path":830,"title":831,"description":11},"\u002Ftutorials\u002Ffront-end\u002Fvue3\u002Fother-api","10. 其他 API",{"path":833,"title":834,"description":11},"\u002Ftutorials\u002Ffront-end\u002Fvue3\u002Fpinia","8. Pinia",{"path":836,"title":837,"description":11},"\u002Ftutorials\u002Ffront-end\u002Fvue3\u002Frouter","7. 路由",{"path":839,"title":840,"description":11},"\u002Ftutorials\u002Ffront-end\u002Fvue3\u002Ftemplate-grammar","3. 指令及模板语法",{"path":842,"title":843,"description":11},"\u002Ftutorials\u002Ffront-end\u002Fvue3\u002Fvue3-new-component","11. Vue3 新组件",{"path":845,"title":846,"description":847},"\u002Ftutorials\u002Ffront-end\u002Fvue3\u002Fwatch","5. 监视","Watch 是 Vue 提供的一个用于监视响应式数据变化并执行相应操作的 API，能够对响应式数据的变化做出一些操作的功能。Vue3 中的 Watch 支持多种用法，包括监视响应式对象、ref 对象、数组、函数等。",{"path":849,"title":850,"description":11},"\u002Ftutorials\u002Fmq\u002Fkafka\u002Fkafka-introduction","Kafka 简介与安装",{"path":852,"title":853,"description":11},"\u002Ftutorials\u002Fmq\u002Fkafka\u002Fkafka-producer-consumer","Kafka 生产者与消费者",{"path":855,"title":856,"description":857},"\u002Ftutorials\u002Fmq\u002Fkafka\u002Fkafka-springboot","Spring Boot 整合 Kafka","Spring Kafka 提供了对 Apache Kafka 的便捷集成。",{"path":859,"title":860,"description":861},"\u002Ftutorials\u002Fmq\u002Frabbitmq\u002Frabbitmq-exchange","RabbitMQ Exchange 详解","Exchange（交换机）是 RabbitMQ 的核心组件，负责接收生产者发送的消息，并根据规则将消息路由到一个或多个队列。",{"path":863,"title":864,"description":11},"\u002Ftutorials\u002Fmq\u002Frabbitmq\u002Frabbitmq-introduction","RabbitMQ 简介与安装",{"path":866,"title":867,"description":868},"\u002Ftutorials\u002Fmq\u002Frabbitmq\u002Frabbitmq-reliability","RabbitMQ 消息可靠性","消息可靠性是消息队列的核心要求，RabbitMQ 提供了多种机制来保证消息不丢失。",{"path":870,"title":871,"description":872},"\u002Ftutorials\u002Fmq\u002Frabbitmq\u002Frabbitmq-springboot","Spring Boot 整合 RabbitMQ","Spring AMQP 提供了对 RabbitMQ 的便捷集成，大大简化了开发工作。",{"path":874,"title":875,"description":11},"\u002Ftutorials\u002Fmq\u002Frocketmq\u002Frocketmq-client","RocketMQ 客户端使用",{"path":877,"title":878,"description":11},"\u002Ftutorials\u002Fmq\u002Frocketmq\u002Frocketmq-concepts","RocketMQ 核心概念",{"path":880,"title":881,"description":11},"\u002Ftutorials\u002Fmq\u002Frocketmq\u002Frocketmq-installation","RocketMQ 安装部署",{"path":883,"title":884,"description":885},"\u002Ftutorials\u002Fmq\u002Frocketmq\u002Frocketmq-message-type","RocketMQ 消息类型","RocketMQ 支持多种消息类型，满足不同业务场景需求。",{"id":887,"title":350,"body":888,"description":351,"extension":3598,"meta":3599,"navigation":1324,"path":349,"seo":3600,"stem":3601,"__hash__":3602},"docs\u002Finterview\u002Fredis.md",{"type":889,"value":890,"toc":3551},"minimark",[891,894,897,902,907,997,999,1003,1064,1134,1136,1140,1200,1206,1208,1212,1279,1358,1360,1364,1368,1439,1445,1447,1451,1487,1497,1499,1503,1508,1518,1523,1541,1543,1547,1552,1560,1565,1573,1589,1591,1595,1599,1604,1618,1623,1638,1640,1644,1649,1654,1659,1664,1669,1674,1692,1694,1698,1703,1715,1720,1736,1779,1781,1785,1789,1845,1850,1901,1906,2029,2031,2035,2040,2045,2056,2061,2069,2074,2082,2087,2136,2138,2142,2147,2181,2186,2218,2223,2234,2236,2240,2244,2315,2325,2327,2331,2336,2347,2352,2372,2377,2407,2409,2413,2477,2479,2483,2543,2549,2551,2555,2559,2612,2618,2620,2624,2629,2637,2642,2690,2692,2696,2700,2710,2715,2776,2778,2782,2788,2793,2807,2813,2815,2819,2874,2876,2880,2949,2951,2955,2959,3331,3333,3337,3405,3407,3411,3414,3457,3460,3505,3508,3547],[892,893,351],"p",{},[895,896],"hr",{},[898,899,901],"h2",{"id":900},"一数据结构","一、数据结构",[903,904,906],"h3",{"id":905},"q1-redis-的五种基本数据类型及底层实现","Q1: Redis 的五种基本数据类型及底层实现",[908,909,910,927],"table",{},[911,912,913],"thead",{},[914,915,916,921,924],"tr",{},[917,918,920],"th",{"align":919},"left","类型",[917,922,923],{"align":919},"底层实现",[917,925,926],{"align":919},"应用场景",[928,929,930,945,958,971,984],"tbody",{},[914,931,932,939,942],{},[933,934,935],"td",{"align":919},[936,937,938],"strong",{},"String",[933,940,941],{"align":919},"SDS（动态字符串）",[933,943,944],{"align":919},"缓存、计数器、分布式锁",[914,946,947,952,955],{},[933,948,949],{"align":919},[936,950,951],{},"Hash",[933,953,954],{"align":919},"ziplist \u002F hashtable",[933,956,957],{"align":919},"对象属性存储",[914,959,960,965,968],{},[933,961,962],{"align":919},[936,963,964],{},"List",[933,966,967],{"align":919},"quicklist（ziplist + 链表）",[933,969,970],{"align":919},"消息队列、时间线",[914,972,973,978,981],{},[933,974,975],{"align":919},[936,976,977],{},"Set",[933,979,980],{"align":919},"intset \u002F hashtable",[933,982,983],{"align":919},"标签、共同好友",[914,985,986,991,994],{},[933,987,988],{"align":919},[936,989,990],{},"ZSet",[933,992,993],{"align":919},"ziplist \u002F skiplist + hashtable",[933,995,996],{"align":919},"排行榜、优先队列",[895,998],{},[903,1000,1002],{"id":1001},"q2-为什么用-sds-而不是-c-字符串","Q2: 为什么用 SDS 而不是 C 字符串？",[908,1004,1005,1018],{},[911,1006,1007],{},[914,1008,1009,1012,1015],{},[917,1010,1011],{"align":919},"特性",[917,1013,1014],{"align":919},"C 字符串",[917,1016,1017],{"align":919},"SDS",[928,1019,1020,1031,1042,1053],{},[914,1021,1022,1025,1028],{},[933,1023,1024],{"align":919},"获取长度",[933,1026,1027],{"align":919},"O(N) 遍历",[933,1029,1030],{"align":919},"O(1) 直接读 len",[914,1032,1033,1036,1039],{},[933,1034,1035],{"align":919},"缓冲区溢出",[933,1037,1038],{"align":919},"可能溢出",[933,1040,1041],{"align":919},"自动扩容",[914,1043,1044,1047,1050],{},[933,1045,1046],{"align":919},"二进制安全",[933,1048,1049],{"align":919},"❌ 不支持 \\0",[933,1051,1052],{"align":919},"✅ 支持任意二进制",[914,1054,1055,1058,1061],{},[933,1056,1057],{"align":919},"内存分配",[933,1059,1060],{"align":919},"每次修改都分配",[933,1062,1063],{"align":919},"空间预分配、惰性释放",[1065,1066,1070],"pre",{"className":1067,"code":1068,"language":1069,"meta":11,"style":11},"language-c shiki shiki-themes github-light github-light github-dark","struct sdshdr {\n    int len;      \u002F\u002F 已用长度\n    int free;     \u002F\u002F 剩余空间\n    char buf[];   \u002F\u002F 实际数据\n};\n","c",[1071,1072,1073,1086,1099,1110,1128],"code",{"__ignoreMap":11},[1074,1075,1078,1082],"span",{"class":1076,"line":1077},"line",1,[1074,1079,1081],{"class":1080},"s8jYJ","struct",[1074,1083,1085],{"class":1084},"sxrX7"," sdshdr {\n",[1074,1087,1089,1092,1095],{"class":1076,"line":1088},2,[1074,1090,1091],{"class":1080},"    int",[1074,1093,1094],{"class":1084}," len;",[1074,1096,1098],{"class":1097},"sCsY4","      \u002F\u002F 已用长度\n",[1074,1100,1102,1104,1107],{"class":1076,"line":1101},3,[1074,1103,1091],{"class":1080},[1074,1105,1106],{"class":1084}," free;",[1074,1108,1109],{"class":1097},"     \u002F\u002F 剩余空间\n",[1074,1111,1113,1116,1119,1122,1125],{"class":1076,"line":1112},4,[1074,1114,1115],{"class":1080},"    char",[1074,1117,1118],{"class":1084}," buf",[1074,1120,1121],{"class":1080},"[]",[1074,1123,1124],{"class":1084},";",[1074,1126,1127],{"class":1097},"   \u002F\u002F 实际数据\n",[1074,1129,1131],{"class":1076,"line":1130},5,[1074,1132,1133],{"class":1084},"};\n",[895,1135],{},[903,1137,1139],{"id":1138},"q3-zset-为什么用跳表而不是红黑树","Q3: ZSet 为什么用跳表而不是红黑树？",[908,1141,1142,1154],{},[911,1143,1144],{},[914,1145,1146,1148,1151],{},[917,1147,1011],{"align":919},[917,1149,1150],{"align":919},"跳表",[917,1152,1153],{"align":919},"红黑树",[928,1155,1156,1167,1178,1189],{},[914,1157,1158,1161,1164],{},[933,1159,1160],{"align":919},"实现复杂度",[933,1162,1163],{"align":919},"简单",[933,1165,1166],{"align":919},"复杂",[914,1168,1169,1172,1175],{},[933,1170,1171],{"align":919},"范围查询",[933,1173,1174],{"align":919},"✅ O(logN + M)",[933,1176,1177],{"align":919},"需要中序遍历",[914,1179,1180,1183,1186],{},[933,1181,1182],{"align":919},"内存占用",[933,1184,1185],{"align":919},"略多",[933,1187,1188],{"align":919},"略少",[914,1190,1191,1194,1197],{},[933,1192,1193],{"align":919},"并发友好",[933,1195,1196],{"align":919},"更容易加锁",[933,1198,1199],{"align":919},"旋转操作复杂",[892,1201,1202,1205],{},[936,1203,1204],{},"结论："," 跳表实现简单，范围查询效率高，更适合 Redis 场景。",[895,1207],{},[903,1209,1211],{"id":1210},"q4-常用的扩展数据类型","Q4: 常用的扩展数据类型",[908,1213,1214,1225],{},[911,1215,1216],{},[914,1217,1218,1220,1223],{},[917,1219,920],{"align":919},[917,1221,1222],{"align":919},"说明",[917,1224,926],{"align":919},[928,1226,1227,1240,1253,1266],{},[914,1228,1229,1234,1237],{},[933,1230,1231],{"align":919},[936,1232,1233],{},"Bitmap",[933,1235,1236],{"align":919},"位图",[933,1238,1239],{"align":919},"用户签到、在线状态",[914,1241,1242,1247,1250],{},[933,1243,1244],{"align":919},[936,1245,1246],{},"HyperLogLog",[933,1248,1249],{"align":919},"基数统计（误差 0.81%）",[933,1251,1252],{"align":919},"UV 统计",[914,1254,1255,1260,1263],{},[933,1256,1257],{"align":919},[936,1258,1259],{},"Geo",[933,1261,1262],{"align":919},"地理位置",[933,1264,1265],{"align":919},"附近的人、门店",[914,1267,1268,1273,1276],{},[933,1269,1270],{"align":919},[936,1271,1272],{},"Stream",[933,1274,1275],{"align":919},"消息流",[933,1277,1278],{"align":919},"消息队列（Redis 5.0+）",[1065,1280,1284],{"className":1281,"code":1282,"language":1283,"meta":11,"style":11},"language-bash shiki shiki-themes github-light github-light github-dark","# Bitmap 签到\nSETBIT user:sign:1001:202401 1 1  # 1月2日签到\nBITCOUNT user:sign:1001:202401    # 统计签到天数\n\n# HyperLogLog UV\nPFADD uv:20240101 user1 user2 user3\nPFCOUNT uv:20240101\n","bash",[1071,1285,1286,1291,1310,1320,1326,1331,1349],{"__ignoreMap":11},[1074,1287,1288],{"class":1076,"line":1077},[1074,1289,1290],{"class":1097},"# Bitmap 签到\n",[1074,1292,1293,1297,1301,1305,1307],{"class":1076,"line":1088},[1074,1294,1296],{"class":1295},"snPdu","SETBIT",[1074,1298,1300],{"class":1299},"sIIMD"," user:sign:1001:202401",[1074,1302,1304],{"class":1303},"sBjJW"," 1",[1074,1306,1304],{"class":1303},[1074,1308,1309],{"class":1097},"  # 1月2日签到\n",[1074,1311,1312,1315,1317],{"class":1076,"line":1101},[1074,1313,1314],{"class":1295},"BITCOUNT",[1074,1316,1300],{"class":1299},[1074,1318,1319],{"class":1097},"    # 统计签到天数\n",[1074,1321,1322],{"class":1076,"line":1112},[1074,1323,1325],{"emptyLinePlaceholder":1324},true,"\n",[1074,1327,1328],{"class":1076,"line":1130},[1074,1329,1330],{"class":1097},"# HyperLogLog UV\n",[1074,1332,1334,1337,1340,1343,1346],{"class":1076,"line":1333},6,[1074,1335,1336],{"class":1295},"PFADD",[1074,1338,1339],{"class":1299}," uv:20240101",[1074,1341,1342],{"class":1299}," user1",[1074,1344,1345],{"class":1299}," user2",[1074,1347,1348],{"class":1299}," user3\n",[1074,1350,1352,1355],{"class":1076,"line":1351},7,[1074,1353,1354],{"class":1295},"PFCOUNT",[1074,1356,1357],{"class":1299}," uv:20240101\n",[895,1359],{},[898,1361,1363],{"id":1362},"二持久化","二、持久化",[903,1365,1367],{"id":1366},"q5-rdb-和-aof-的区别","Q5: RDB 和 AOF 的区别",[908,1369,1370,1382],{},[911,1371,1372],{},[914,1373,1374,1376,1379],{},[917,1375,1011],{"align":919},[917,1377,1378],{"align":919},"RDB",[917,1380,1381],{"align":919},"AOF",[928,1383,1384,1395,1406,1417,1428],{},[914,1385,1386,1389,1392],{},[933,1387,1388],{"align":919},"原理",[933,1390,1391],{"align":919},"内存快照",[933,1393,1394],{"align":919},"命令日志",[914,1396,1397,1400,1403],{},[933,1398,1399],{"align":919},"文件大小",[933,1401,1402],{"align":919},"小（二进制压缩）",[933,1404,1405],{"align":919},"大（文本命令）",[914,1407,1408,1411,1414],{},[933,1409,1410],{"align":919},"恢复速度",[933,1412,1413],{"align":919},"快",[933,1415,1416],{"align":919},"慢",[914,1418,1419,1422,1425],{},[933,1420,1421],{"align":919},"数据安全",[933,1423,1424],{"align":919},"可能丢失分钟级数据",[933,1426,1427],{"align":919},"最多丢 1 秒",[914,1429,1430,1433,1436],{},[933,1431,1432],{"align":919},"性能影响",[933,1434,1435],{"align":919},"fork 子进程时影响",[933,1437,1438],{"align":919},"everysec 影响小",[892,1440,1441,1444],{},[936,1442,1443],{},"推荐："," 同时开启 RDB + AOF，优先用 AOF 恢复。",[895,1446],{},[903,1448,1450],{"id":1449},"q6-aof-的同步策略","Q6: AOF 的同步策略",[1065,1452,1454],{"className":1281,"code":1453,"language":1283,"meta":11,"style":11},"appendfsync always    # 每条命令都刷盘，最安全，性能最差\nappendfsync everysec  # 每秒刷盘（默认），折中方案\nappendfsync no        # 不主动刷盘，由 OS 决定，性能最好，风险最大\n",[1071,1455,1456,1467,1477],{"__ignoreMap":11},[1074,1457,1458,1461,1464],{"class":1076,"line":1077},[1074,1459,1460],{"class":1295},"appendfsync",[1074,1462,1463],{"class":1299}," always",[1074,1465,1466],{"class":1097},"    # 每条命令都刷盘，最安全，性能最差\n",[1074,1468,1469,1471,1474],{"class":1076,"line":1088},[1074,1470,1460],{"class":1295},[1074,1472,1473],{"class":1299}," everysec",[1074,1475,1476],{"class":1097},"  # 每秒刷盘（默认），折中方案\n",[1074,1478,1479,1481,1484],{"class":1076,"line":1101},[1074,1480,1460],{"class":1295},[1074,1482,1483],{"class":1299}," no",[1074,1485,1486],{"class":1097},"        # 不主动刷盘，由 OS 决定，性能最好，风险最大\n",[892,1488,1489,1492,1493,1496],{},[936,1490,1491],{},"生产建议："," 使用 ",[1071,1494,1495],{},"everysec","，最多丢失 1 秒数据。",[895,1498],{},[903,1500,1502],{"id":1501},"q7-aof-重写机制","Q7: AOF 重写机制",[892,1504,1505],{},[936,1506,1507],{},"为什么需要重写？",[1509,1510,1511,1515],"ul",{},[1512,1513,1514],"li",{},"AOF 文件持续增长",[1512,1516,1517],{},"很多命令可以合并（如多次 INCR 合并为 SET）",[892,1519,1520],{},[936,1521,1522],{},"重写流程：",[1524,1525,1526,1529,1532,1535,1538],"ol",{},[1512,1527,1528],{},"fork 子进程，读取当前内存数据",[1512,1530,1531],{},"子进程写入新 AOF 文件",[1512,1533,1534],{},"主进程继续处理命令，增量写入 AOF 重写缓冲区",[1512,1536,1537],{},"子进程完成后，将缓冲区内容追加到新文件",[1512,1539,1540],{},"原子替换旧 AOF 文件",[895,1542],{},[903,1544,1546],{"id":1545},"q8-混合持久化","Q8: 混合持久化",[892,1548,1549],{},[936,1550,1551],{},"原理（Redis 4.0+）：",[1509,1553,1554,1557],{},[1512,1555,1556],{},"AOF 重写时，先写入 RDB 格式快照",[1512,1558,1559],{},"快照后的增量命令以 AOF 格式追加",[892,1561,1562],{},[936,1563,1564],{},"优点：",[1509,1566,1567,1570],{},[1512,1568,1569],{},"恢复速度快（RDB 部分）",[1512,1571,1572],{},"数据安全性高（AOF 部分）",[1065,1574,1576],{"className":1281,"code":1575,"language":1283,"meta":11,"style":11},"aof-use-rdb-preamble yes  # 开启混合持久化\n",[1071,1577,1578],{"__ignoreMap":11},[1074,1579,1580,1583,1586],{"class":1076,"line":1077},[1074,1581,1582],{"class":1295},"aof-use-rdb-preamble",[1074,1584,1585],{"class":1299}," yes",[1074,1587,1588],{"class":1097},"  # 开启混合持久化\n",[895,1590],{},[898,1592,1594],{"id":1593},"三集群与高可用","三、集群与高可用",[903,1596,1598],{"id":1597},"q9-主从复制原理","Q9: 主从复制原理",[892,1600,1601],{},[936,1602,1603],{},"全量复制：",[1524,1605,1606,1609,1612,1615],{},[1512,1607,1608],{},"Slave 发送 PSYNC 命令",[1512,1610,1611],{},"Master 执行 BGSAVE，生成 RDB",[1512,1613,1614],{},"Master 发送 RDB 给 Slave",[1512,1616,1617],{},"Master 发送复制缓冲区内容",[892,1619,1620],{},[936,1621,1622],{},"增量复制（Redis 2.8+）：",[1509,1624,1625,1632,1635],{},[1512,1626,1627,1628,1631],{},"使用 ",[1071,1629,1630],{},"repl_backlog"," 环形缓冲区",[1512,1633,1634],{},"Slave 断线重连后，发送 offset",[1512,1636,1637],{},"如果 offset 在 backlog 内，只同步增量",[895,1639],{},[903,1641,1643],{"id":1642},"q10-哨兵模式的选主流程","Q10: 哨兵模式的选主流程",[892,1645,1646],{},[936,1647,1648],{},"1. 主观下线（SDOWN）：",[1509,1650,1651],{},[1512,1652,1653],{},"Sentinel 认为 Master 不可用",[892,1655,1656],{},[936,1657,1658],{},"2. 客观下线（ODOWN）：",[1509,1660,1661],{},[1512,1662,1663],{},"超过 quorum 个 Sentinel 都认为 Master 不可用",[892,1665,1666],{},[936,1667,1668],{},"3. 选举 Sentinel Leader（Raft）",[892,1670,1671],{},[936,1672,1673],{},"4. Sentinel Leader 选新 Master：",[1524,1675,1676,1679,1686,1689],{},[1512,1677,1678],{},"过滤不健康的 Slave",[1512,1680,1681,1682,1685],{},"按优先级 ",[1071,1683,1684],{},"slave-priority"," 排序",[1512,1687,1688],{},"优先级相同，选复制偏移量最大的",[1512,1690,1691],{},"偏移量相同，选 runid 最小的",[895,1693],{},[903,1695,1697],{"id":1696},"q11-cluster-模式的特点","Q11: Cluster 模式的特点",[892,1699,1700],{},[936,1701,1702],{},"槽位分配：",[1509,1704,1705,1708],{},[1512,1706,1707],{},"固定 16384 个槽位",[1512,1709,1710,1711,1714],{},"数据按 ",[1071,1712,1713],{},"CRC16(key) % 16384"," 分配到槽",[892,1716,1717],{},[936,1718,1719],{},"多 key 操作限制：",[1509,1721,1722,1725],{},[1512,1723,1724],{},"多 key 命令（MGET、MSET）必须在同一槽",[1512,1726,1727,1728,1731,1732,1735],{},"使用 hash tag：",[1071,1729,1730],{},"{user}:1001"," 和 ",[1071,1733,1734],{},"{user}:1002"," 在同一槽",[1065,1737,1739],{"className":1281,"code":1738,"language":1283,"meta":11,"style":11},"# hash tag 用法\nSET {order}:1001 value1\nSET {order}:1002 value2\nMGET {order}:1001 {order}:1002  # ✅ 可以\n",[1071,1740,1741,1746,1757,1767],{"__ignoreMap":11},[1074,1742,1743],{"class":1076,"line":1077},[1074,1744,1745],{"class":1097},"# hash tag 用法\n",[1074,1747,1748,1751,1754],{"class":1076,"line":1088},[1074,1749,1750],{"class":1295},"SET",[1074,1752,1753],{"class":1299}," {order}:1001",[1074,1755,1756],{"class":1299}," value1\n",[1074,1758,1759,1761,1764],{"class":1076,"line":1101},[1074,1760,1750],{"class":1295},[1074,1762,1763],{"class":1299}," {order}:1002",[1074,1765,1766],{"class":1299}," value2\n",[1074,1768,1769,1772,1774,1776],{"class":1076,"line":1112},[1074,1770,1771],{"class":1295},"MGET",[1074,1773,1753],{"class":1299},[1074,1775,1763],{"class":1299},[1074,1777,1778],{"class":1097},"  # ✅ 可以\n",[895,1780],{},[898,1782,1784],{"id":1783},"四缓存问题","四、缓存问题",[903,1786,1788],{"id":1787},"q12-缓存穿透击穿雪崩","Q12: 缓存穿透、击穿、雪崩",[908,1790,1791,1804],{},[911,1792,1793],{},[914,1794,1795,1798,1801],{},[917,1796,1797],{"align":919},"问题",[917,1799,1800],{"align":919},"原因",[917,1802,1803],{"align":919},"解决方案",[928,1805,1806,1819,1832],{},[914,1807,1808,1813,1816],{},[933,1809,1810],{"align":919},[936,1811,1812],{},"穿透",[933,1814,1815],{"align":919},"查询不存在的数据",[933,1817,1818],{"align":919},"布隆过滤器、缓存空值",[914,1820,1821,1826,1829],{},[933,1822,1823],{"align":919},[936,1824,1825],{},"击穿",[933,1827,1828],{"align":919},"热点 key 过期",[933,1830,1831],{"align":919},"互斥锁、永不过期 + 异步更新",[914,1833,1834,1839,1842],{},[933,1835,1836],{"align":919},[936,1837,1838],{},"雪崩",[933,1840,1841],{"align":919},"大量 key 同时过期",[933,1843,1844],{"align":919},"随机过期时间、多级缓存",[892,1846,1847],{},[936,1848,1849],{},"缓存穿透 - 布隆过滤器：",[1065,1851,1855],{"className":1852,"code":1853,"language":1854,"meta":11,"style":11},"language-python shiki shiki-themes github-light github-light github-dark","# 查询前先判断\nif not bloom_filter.exists(key):\n    return None  # 一定不存在\ndata = cache.get(key) or db.get(key)\n","python",[1071,1856,1857,1862,1873,1884],{"__ignoreMap":11},[1074,1858,1859],{"class":1076,"line":1077},[1074,1860,1861],{"class":1097},"# 查询前先判断\n",[1074,1863,1864,1867,1870],{"class":1076,"line":1088},[1074,1865,1866],{"class":1080},"if",[1074,1868,1869],{"class":1080}," not",[1074,1871,1872],{"class":1084}," bloom_filter.exists(key):\n",[1074,1874,1875,1878,1881],{"class":1076,"line":1101},[1074,1876,1877],{"class":1080},"    return",[1074,1879,1880],{"class":1303}," None",[1074,1882,1883],{"class":1097},"  # 一定不存在\n",[1074,1885,1886,1889,1892,1895,1898],{"class":1076,"line":1112},[1074,1887,1888],{"class":1084},"data ",[1074,1890,1891],{"class":1080},"=",[1074,1893,1894],{"class":1084}," cache.get(key) ",[1074,1896,1897],{"class":1080},"or",[1074,1899,1900],{"class":1084}," db.get(key)\n",[892,1902,1903],{},[936,1904,1905],{},"缓存击穿 - 互斥锁：",[1065,1907,1909],{"className":1852,"code":1908,"language":1854,"meta":11,"style":11},"def get_data(key):\n    data = cache.get(key)\n    if data:\n        return data\n    \n    if acquire_lock(key):\n        try:\n            data = cache.get(key)  # 双重检查\n            if not data:\n                data = db.get(key)\n                cache.set(key, data, ttl)\n        finally:\n            release_lock(key)\n    return data\n",[1071,1910,1911,1922,1932,1940,1948,1953,1960,1968,1982,1992,2002,2008,2016,2022],{"__ignoreMap":11},[1074,1912,1913,1916,1919],{"class":1076,"line":1077},[1074,1914,1915],{"class":1080},"def",[1074,1917,1918],{"class":1295}," get_data",[1074,1920,1921],{"class":1084},"(key):\n",[1074,1923,1924,1927,1929],{"class":1076,"line":1088},[1074,1925,1926],{"class":1084},"    data ",[1074,1928,1891],{"class":1080},[1074,1930,1931],{"class":1084}," cache.get(key)\n",[1074,1933,1934,1937],{"class":1076,"line":1101},[1074,1935,1936],{"class":1080},"    if",[1074,1938,1939],{"class":1084}," data:\n",[1074,1941,1942,1945],{"class":1076,"line":1112},[1074,1943,1944],{"class":1080},"        return",[1074,1946,1947],{"class":1084}," data\n",[1074,1949,1950],{"class":1076,"line":1130},[1074,1951,1952],{"class":1084},"    \n",[1074,1954,1955,1957],{"class":1076,"line":1333},[1074,1956,1936],{"class":1080},[1074,1958,1959],{"class":1084}," acquire_lock(key):\n",[1074,1961,1962,1965],{"class":1076,"line":1351},[1074,1963,1964],{"class":1080},"        try",[1074,1966,1967],{"class":1084},":\n",[1074,1969,1971,1974,1976,1979],{"class":1076,"line":1970},8,[1074,1972,1973],{"class":1084},"            data ",[1074,1975,1891],{"class":1080},[1074,1977,1978],{"class":1084}," cache.get(key)  ",[1074,1980,1981],{"class":1097},"# 双重检查\n",[1074,1983,1985,1988,1990],{"class":1076,"line":1984},9,[1074,1986,1987],{"class":1080},"            if",[1074,1989,1869],{"class":1080},[1074,1991,1939],{"class":1084},[1074,1993,1995,1998,2000],{"class":1076,"line":1994},10,[1074,1996,1997],{"class":1084},"                data ",[1074,1999,1891],{"class":1080},[1074,2001,1900],{"class":1084},[1074,2003,2005],{"class":1076,"line":2004},11,[1074,2006,2007],{"class":1084},"                cache.set(key, data, ttl)\n",[1074,2009,2011,2014],{"class":1076,"line":2010},12,[1074,2012,2013],{"class":1080},"        finally",[1074,2015,1967],{"class":1084},[1074,2017,2019],{"class":1076,"line":2018},13,[1074,2020,2021],{"class":1084},"            release_lock(key)\n",[1074,2023,2025,2027],{"class":1076,"line":2024},14,[1074,2026,1877],{"class":1080},[1074,2028,1947],{"class":1084},[895,2030],{},[903,2032,2034],{"id":2033},"q13-缓存与数据库一致性","Q13: 缓存与数据库一致性",[892,2036,2037],{},[936,2038,2039],{},"Cache Aside 模式（推荐）：",[892,2041,2042],{},[936,2043,2044],{},"读流程：",[1524,2046,2047,2050,2053],{},[1512,2048,2049],{},"先读缓存",[1512,2051,2052],{},"缓存命中则返回",[1512,2054,2055],{},"未命中则查 DB，写入缓存",[892,2057,2058],{},[936,2059,2060],{},"写流程：",[1524,2062,2063,2066],{},[1512,2064,2065],{},"更新 DB",[1512,2067,2068],{},"删除缓存（而非更新）",[892,2070,2071],{},[936,2072,2073],{},"为什么删除而不是更新？",[1509,2075,2076,2079],{},[1512,2077,2078],{},"避免并发写导致缓存和 DB 不一致",[1512,2080,2081],{},"延迟更新，下次读取时自动加载",[892,2083,2084],{},[936,2085,2086],{},"延迟双删：",[1065,2088,2090],{"className":1852,"code":2089,"language":1854,"meta":11,"style":11},"def update_data(key, value):\n    cache.delete(key)         # 第一次删除\n    db.update(key, value)\n    time.sleep(0.5)           # 等待可能的并发读\n    cache.delete(key)         # 第二次删除\n",[1071,2091,2092,2102,2110,2115,2129],{"__ignoreMap":11},[1074,2093,2094,2096,2099],{"class":1076,"line":1077},[1074,2095,1915],{"class":1080},[1074,2097,2098],{"class":1295}," update_data",[1074,2100,2101],{"class":1084},"(key, value):\n",[1074,2103,2104,2107],{"class":1076,"line":1088},[1074,2105,2106],{"class":1084},"    cache.delete(key)         ",[1074,2108,2109],{"class":1097},"# 第一次删除\n",[1074,2111,2112],{"class":1076,"line":1101},[1074,2113,2114],{"class":1084},"    db.update(key, value)\n",[1074,2116,2117,2120,2123,2126],{"class":1076,"line":1112},[1074,2118,2119],{"class":1084},"    time.sleep(",[1074,2121,2122],{"class":1303},"0.5",[1074,2124,2125],{"class":1084},")           ",[1074,2127,2128],{"class":1097},"# 等待可能的并发读\n",[1074,2130,2131,2133],{"class":1076,"line":1130},[1074,2132,2106],{"class":1084},[1074,2134,2135],{"class":1097},"# 第二次删除\n",[895,2137],{},[903,2139,2141],{"id":2140},"q14-分布式锁的正确实现","Q14: 分布式锁的正确实现",[892,2143,2144],{},[936,2145,2146],{},"加锁：",[1065,2148,2150],{"className":1281,"code":2149,"language":1283,"meta":11,"style":11},"SET lock_key unique_value NX PX 30000\n# NX: 不存在才设置\n# PX: 30 秒过期\n",[1071,2151,2152,2171,2176],{"__ignoreMap":11},[1074,2153,2154,2156,2159,2162,2165,2168],{"class":1076,"line":1077},[1074,2155,1750],{"class":1295},[1074,2157,2158],{"class":1299}," lock_key",[1074,2160,2161],{"class":1299}," unique_value",[1074,2163,2164],{"class":1299}," NX",[1074,2166,2167],{"class":1299}," PX",[1074,2169,2170],{"class":1303}," 30000\n",[1074,2172,2173],{"class":1076,"line":1088},[1074,2174,2175],{"class":1097},"# NX: 不存在才设置\n",[1074,2177,2178],{"class":1076,"line":1101},[1074,2179,2180],{"class":1097},"# PX: 30 秒过期\n",[892,2182,2183],{},[936,2184,2185],{},"解锁（Lua 保证原子性）：",[1065,2187,2191],{"className":2188,"code":2189,"language":2190,"meta":11,"style":11},"language-lua shiki shiki-themes github-light github-light github-dark","if redis.call('get', KEYS[1]) == ARGV[1] then\n    return redis.call('del', KEYS[1])\nelse\n    return 0\nend\n","lua",[1071,2192,2193,2198,2203,2208,2213],{"__ignoreMap":11},[1074,2194,2195],{"class":1076,"line":1077},[1074,2196,2197],{},"if redis.call('get', KEYS[1]) == ARGV[1] then\n",[1074,2199,2200],{"class":1076,"line":1088},[1074,2201,2202],{},"    return redis.call('del', KEYS[1])\n",[1074,2204,2205],{"class":1076,"line":1101},[1074,2206,2207],{},"else\n",[1074,2209,2210],{"class":1076,"line":1112},[1074,2211,2212],{},"    return 0\n",[1074,2214,2215],{"class":1076,"line":1130},[1074,2216,2217],{},"end\n",[892,2219,2220],{},[936,2221,2222],{},"锁续期（看门狗）：",[1509,2224,2225,2228,2231],{},[1512,2226,2227],{},"后台线程定期检查锁",[1512,2229,2230],{},"业务未完成则延长过期时间",[1512,2232,2233],{},"Redisson 已内置实现",[895,2235],{},[898,2237,2239],{"id":2238},"五性能优化","五、性能优化",[903,2241,2243],{"id":2242},"q15-内存淘汰策略","Q15: 内存淘汰策略",[908,2245,2246,2258],{},[911,2247,2248],{},[914,2249,2250,2253,2255],{},[917,2251,2252],{"align":919},"策略",[917,2254,1222],{"align":919},[917,2256,2257],{"align":919},"适用场景",[928,2259,2260,2271,2282,2293,2304],{},[914,2261,2262,2265,2268],{},[933,2263,2264],{"align":919},"noeviction",[933,2266,2267],{"align":919},"不淘汰，写入报错",[933,2269,2270],{"align":919},"数据不能丢",[914,2272,2273,2276,2279],{},[933,2274,2275],{"align":919},"allkeys-lru",[933,2277,2278],{"align":919},"淘汰最近最少使用",[933,2280,2281],{"align":919},"通用缓存",[914,2283,2284,2287,2290],{},[933,2285,2286],{"align":919},"allkeys-lfu",[933,2288,2289],{"align":919},"淘汰最不常用（Redis 4.0+）",[933,2291,2292],{"align":919},"热点数据明显",[914,2294,2295,2298,2301],{},[933,2296,2297],{"align":919},"volatile-ttl",[933,2299,2300],{"align":919},"淘汰快过期的",[933,2302,2303],{"align":919},"有过期时间的数据",[914,2305,2306,2309,2312],{},[933,2307,2308],{"align":919},"volatile-lru",[933,2310,2311],{"align":919},"只在设置过期时间的 key 中淘汰",[933,2313,2314],{"align":919},"部分数据需持久化",[892,2316,2317,2319,2320,2322,2323],{},[936,2318,1443],{}," ",[1071,2321,2275],{}," 或 ",[1071,2324,2286],{},[895,2326],{},[903,2328,2330],{"id":2329},"q16-大-key-问题及处理","Q16: 大 key 问题及处理",[892,2332,2333],{},[936,2334,2335],{},"风险：",[1509,2337,2338,2341,2344],{},[1512,2339,2340],{},"阻塞其他请求（单线程）",[1512,2342,2343],{},"网络传输慢",[1512,2345,2346],{},"DEL 可能阻塞主线程",[892,2348,2349],{},[936,2350,2351],{},"发现大 key：",[1065,2353,2355],{"className":1281,"code":2354,"language":1283,"meta":11,"style":11},"redis-cli --bigkeys\nredis-cli --memkeys\n",[1071,2356,2357,2365],{"__ignoreMap":11},[1074,2358,2359,2362],{"class":1076,"line":1077},[1074,2360,2361],{"class":1295},"redis-cli",[1074,2363,2364],{"class":1303}," --bigkeys\n",[1074,2366,2367,2369],{"class":1076,"line":1088},[1074,2368,2361],{"class":1295},[1074,2370,2371],{"class":1303}," --memkeys\n",[892,2373,2374],{},[936,2375,2376],{},"处理方案：",[1524,2378,2379,2385,2398],{},[1512,2380,2381,2384],{},[936,2382,2383],{},"拆分","：大 Hash 拆成多个小 Hash",[1512,2386,2387,2390,2391,2394,2395],{},[936,2388,2389],{},"异步删除","：",[1071,2392,2393],{},"UNLINK"," 代替 ",[1071,2396,2397],{},"DEL",[1512,2399,2400,2390,2403,2406],{},[936,2401,2402],{},"分批操作",[1071,2404,2405],{},"HSCAN\u002FSSCAN"," 分批读取",[895,2408],{},[903,2410,2412],{"id":2411},"q17-redis-为什么这么快","Q17: Redis 为什么这么快",[908,2414,2415,2424],{},[911,2416,2417],{},[914,2418,2419,2422],{},[917,2420,2421],{"align":919},"因素",[917,2423,1222],{"align":919},[928,2425,2426,2436,2446,2456,2466],{},[914,2427,2428,2433],{},[933,2429,2430],{"align":919},[936,2431,2432],{},"内存存储",[933,2434,2435],{"align":919},"直接操作内存，无磁盘 IO",[914,2437,2438,2443],{},[933,2439,2440],{"align":919},[936,2441,2442],{},"单线程命令处理",[933,2444,2445],{"align":919},"无线程切换、无锁竞争",[914,2447,2448,2453],{},[933,2449,2450],{"align":919},[936,2451,2452],{},"IO 多路复用",[933,2454,2455],{"align":919},"epoll 高效处理并发连接",[914,2457,2458,2463],{},[933,2459,2460],{"align":919},[936,2461,2462],{},"高效数据结构",[933,2464,2465],{"align":919},"SDS、跳表、哈希表等",[914,2467,2468,2474],{},[933,2469,2470,2473],{"align":919},[936,2471,2472],{},"IO 多线程","（Redis 6.0+）",[933,2475,2476],{"align":919},"读写网络 IO 多线程",[895,2478],{},[903,2480,2482],{"id":2481},"q18-pipeline-和事务的区别","Q18: Pipeline 和事务的区别",[908,2484,2485,2497],{},[911,2486,2487],{},[914,2488,2489,2491,2494],{},[917,2490,1011],{"align":919},[917,2492,2493],{"align":919},"Pipeline",[917,2495,2496],{"align":919},"事务 (MULTI\u002FEXEC)",[928,2498,2499,2510,2521,2532],{},[914,2500,2501,2504,2507],{},[933,2502,2503],{"align":919},"目的",[933,2505,2506],{"align":919},"减少网络往返",[933,2508,2509],{"align":919},"保证原子执行",[914,2511,2512,2515,2518],{},[933,2513,2514],{"align":919},"原子性",[933,2516,2517],{"align":919},"❌",[933,2519,2520],{"align":919},"✅（部分原子）",[914,2522,2523,2526,2529],{},[933,2524,2525],{"align":919},"回滚",[933,2527,2528],{"align":919},"-",[933,2530,2531],{"align":919},"❌ 不支持",[914,2533,2534,2537,2540],{},[933,2535,2536],{"align":919},"性能",[933,2538,2539],{"align":919},"更好",[933,2541,2542],{"align":919},"略差",[892,2544,2545,2548],{},[936,2546,2547],{},"注意："," Redis 事务不保证 ACID，编译错误会取消事务，运行时错误不回滚。",[895,2550],{},[898,2552,2554],{"id":2553},"六更多八股文","六、更多八股文",[903,2556,2558],{"id":2557},"q19-过期删除策略","Q19: 过期删除策略",[908,2560,2561,2572],{},[911,2562,2563],{},[914,2564,2565,2567,2569],{},[917,2566,2252],{"align":919},[917,2568,1222],{"align":919},[917,2570,2571],{"align":919},"使用场景",[928,2573,2574,2587,2600],{},[914,2575,2576,2581,2584],{},[933,2577,2578],{"align":919},[936,2579,2580],{},"定时删除",[933,2582,2583],{"align":919},"key 过期立即删除",[933,2585,2586],{"align":919},"CPU 开销大，不使用",[914,2588,2589,2594,2597],{},[933,2590,2591],{"align":919},[936,2592,2593],{},"惰性删除",[933,2595,2596],{"align":919},"访问时检查是否过期",[933,2598,2599],{"align":919},"Redis 使用",[914,2601,2602,2607,2610],{},[933,2603,2604],{"align":919},[936,2605,2606],{},"定期删除",[933,2608,2609],{"align":919},"周期性随机检查删除",[933,2611,2599],{"align":919},[892,2613,2614,2617],{},[936,2615,2616],{},"Redis 采用："," 惰性删除 + 定期删除",[895,2619],{},[903,2621,2623],{"id":2622},"q20-热点-key-问题","Q20: 热点 Key 问题",[892,2625,2626],{},[936,2627,2628],{},"危害：",[1509,2630,2631,2634],{},[1512,2632,2633],{},"单节点压力过大",[1512,2635,2636],{},"网络带宽瓶颈",[892,2638,2639],{},[936,2640,2641],{},"解决方案：",[908,2643,2644,2653],{},[911,2645,2646],{},[914,2647,2648,2651],{},[917,2649,2650],{"align":919},"方案",[917,2652,1222],{"align":919},[928,2654,2655,2663,2671,2679],{},[914,2656,2657,2660],{},[933,2658,2659],{"align":919},"本地缓存",[933,2661,2662],{"align":919},"减少 Redis 访问",[914,2664,2665,2668],{},[933,2666,2667],{"align":919},"读写分离",[933,2669,2670],{"align":919},"多个从节点分担读压力",[914,2672,2673,2676],{},[933,2674,2675],{"align":919},"二级 key",[933,2677,2678],{"align":919},"将 key 拆分为多个",[914,2680,2681,2684],{},[933,2682,2683],{"align":919},"热点发现",[933,2685,2686,2689],{"align":919},[1071,2687,2688],{},"--hotkeys"," 参数",[895,2691],{},[903,2693,2695],{"id":2694},"q21-redis-lua-脚本","Q21: Redis Lua 脚本",[892,2697,2698],{},[936,2699,1564],{},[1509,2701,2702,2705,2707],{},[1512,2703,2704],{},"原子性执行",[1512,2706,2506],{},[1512,2708,2709],{},"复杂逻辑封装",[892,2711,2712],{},[936,2713,2714],{},"示例：",[1065,2716,2718],{"className":2188,"code":2717,"language":2190,"meta":11,"style":11},"-- 限流脚本\nlocal key = KEYS[1]\nlocal limit = tonumber(ARGV[1])\nlocal current = tonumber(redis.call('GET', key) or \"0\")\n\nif current + 1 > limit then\n    return 0\nelse\n    redis.call('INCR', key)\n    redis.call('EXPIRE', key, 1)\n    return 1\nend\n",[1071,2719,2720,2725,2730,2735,2740,2744,2749,2753,2757,2762,2767,2772],{"__ignoreMap":11},[1074,2721,2722],{"class":1076,"line":1077},[1074,2723,2724],{},"-- 限流脚本\n",[1074,2726,2727],{"class":1076,"line":1088},[1074,2728,2729],{},"local key = KEYS[1]\n",[1074,2731,2732],{"class":1076,"line":1101},[1074,2733,2734],{},"local limit = tonumber(ARGV[1])\n",[1074,2736,2737],{"class":1076,"line":1112},[1074,2738,2739],{},"local current = tonumber(redis.call('GET', key) or \"0\")\n",[1074,2741,2742],{"class":1076,"line":1130},[1074,2743,1325],{"emptyLinePlaceholder":1324},[1074,2745,2746],{"class":1076,"line":1333},[1074,2747,2748],{},"if current + 1 > limit then\n",[1074,2750,2751],{"class":1076,"line":1351},[1074,2752,2212],{},[1074,2754,2755],{"class":1076,"line":1970},[1074,2756,2207],{},[1074,2758,2759],{"class":1076,"line":1984},[1074,2760,2761],{},"    redis.call('INCR', key)\n",[1074,2763,2764],{"class":1076,"line":1994},[1074,2765,2766],{},"    redis.call('EXPIRE', key, 1)\n",[1074,2768,2769],{"class":1076,"line":2004},[1074,2770,2771],{},"    return 1\n",[1074,2773,2774],{"class":1076,"line":2010},[1074,2775,2217],{},[895,2777],{},[903,2779,2781],{"id":2780},"q22-redlock-分布式锁","Q22: Redlock 分布式锁",[892,2783,2784,2787],{},[936,2785,2786],{},"原理："," 向多个独立 Redis 节点加锁",[892,2789,2790],{},[936,2791,2792],{},"步骤：",[1524,2794,2795,2798,2801,2804],{},[1512,2796,2797],{},"获取当前时间",[1512,2799,2800],{},"依次向 N 个节点请求加锁",[1512,2802,2803],{},"超过半数成功且耗时小于锁过期时间 → 加锁成功",[1512,2805,2806],{},"失败则释放所有节点的锁",[892,2808,2809,2812],{},[936,2810,2811],{},"争议："," Martin Kleppmann 质疑其正确性（时钟漂移问题）",[895,2814],{},[903,2816,2818],{"id":2817},"q23-redis-集群扩容流程","Q23: Redis 集群扩容流程",[1065,2820,2822],{"className":1281,"code":2821,"language":1283,"meta":11,"style":11},"# 1. 添加新节点\nredis-cli --cluster add-node new_host:port existing_host:port\n\n# 2. 重新分配槽位\nredis-cli --cluster reshard existing_host:port\n\n# 3. 数据迁移（自动）\n",[1071,2823,2824,2829,2845,2849,2854,2865,2869],{"__ignoreMap":11},[1074,2825,2826],{"class":1076,"line":1077},[1074,2827,2828],{"class":1097},"# 1. 添加新节点\n",[1074,2830,2831,2833,2836,2839,2842],{"class":1076,"line":1088},[1074,2832,2361],{"class":1295},[1074,2834,2835],{"class":1303}," --cluster",[1074,2837,2838],{"class":1299}," add-node",[1074,2840,2841],{"class":1299}," new_host:port",[1074,2843,2844],{"class":1299}," existing_host:port\n",[1074,2846,2847],{"class":1076,"line":1101},[1074,2848,1325],{"emptyLinePlaceholder":1324},[1074,2850,2851],{"class":1076,"line":1112},[1074,2852,2853],{"class":1097},"# 2. 重新分配槽位\n",[1074,2855,2856,2858,2860,2863],{"class":1076,"line":1130},[1074,2857,2361],{"class":1295},[1074,2859,2835],{"class":1303},[1074,2861,2862],{"class":1299}," reshard",[1074,2864,2844],{"class":1299},[1074,2866,2867],{"class":1076,"line":1333},[1074,2868,1325],{"emptyLinePlaceholder":1324},[1074,2870,2871],{"class":1076,"line":1351},[1074,2872,2873],{"class":1097},"# 3. 数据迁移（自动）\n",[895,2875],{},[903,2877,2879],{"id":2878},"q24-redis-和-memcached-的区别","Q24: Redis 和 Memcached 的区别",[908,2881,2882,2894],{},[911,2883,2884],{},[914,2885,2886,2888,2891],{},[917,2887,1011],{"align":919},[917,2889,2890],{"align":919},"Redis",[917,2892,2893],{"align":919},"Memcached",[928,2895,2896,2907,2917,2928,2938],{},[914,2897,2898,2901,2904],{},[933,2899,2900],{"align":919},"数据类型",[933,2902,2903],{"align":919},"丰富（5 种+）",[933,2905,2906],{"align":919},"只有 String",[914,2908,2909,2912,2915],{},[933,2910,2911],{"align":919},"持久化",[933,2913,2914],{"align":919},"✅",[933,2916,2517],{"align":919},[914,2918,2919,2922,2925],{},[933,2920,2921],{"align":919},"集群",[933,2923,2924],{"align":919},"✅ 原生支持",[933,2926,2927],{"align":919},"客户端分片",[914,2929,2930,2933,2936],{},[933,2931,2932],{"align":919},"多线程",[933,2934,2935],{"align":919},"6.0+ IO 多线程",[933,2937,2932],{"align":919},[914,2939,2940,2943,2946],{},[933,2941,2942],{"align":919},"内存效率",[933,2944,2945],{"align":919},"略低",[933,2947,2948],{"align":919},"更高",[895,2950],{},[898,2952,2954],{"id":2953},"七常用命令","七、常用命令",[903,2956,2958],{"id":2957},"q25-redis-各数据类型常用命令汇总","Q25: Redis 各数据类型常用命令汇总",[908,2960,2961,2973],{},[911,2962,2963],{},[914,2964,2965,2968,2971],{},[917,2966,2967],{"align":919},"分类",[917,2969,2970],{"align":919},"命令",[917,2972,1222],{"align":919},[928,2974,2975,2992,3007,3019,3031,3045,3062,3077,3089,3101,3118,3133,3145,3160,3177,3189,3201,3216,3230,3242,3254,3266,3280,3295,3307,3319],{},[914,2976,2977,2981,2989],{},[933,2978,2979],{"align":919},[936,2980,938],{},[933,2982,2983,2985,2986],{"align":919},[1071,2984,1750],{}," \u002F ",[1071,2987,2988],{},"GET",[933,2990,2991],{"align":919},"设置\u002F获取值",[914,2993,2994,2996,3004],{},[933,2995],{"align":919},[933,2997,2998,2985,3001],{"align":919},[1071,2999,3000],{},"INCR",[1071,3002,3003],{},"DECR",[933,3005,3006],{"align":919},"自增\u002F自减",[914,3008,3009,3011,3016],{},[933,3010],{"align":919},[933,3012,3013],{"align":919},[1071,3014,3015],{},"SETNX",[933,3017,3018],{"align":919},"Key 不存在才设置（分布式锁核心）",[914,3020,3021,3023,3028],{},[933,3022],{"align":919},[933,3024,3025],{"align":919},[1071,3026,3027],{},"SETEX",[933,3029,3030],{"align":919},"设置值并指定过期时间（秒）",[914,3032,3033,3035,3042],{},[933,3034],{"align":919},[933,3036,3037,2985,3040],{"align":919},[1071,3038,3039],{},"MSET",[1071,3041,1771],{},[933,3043,3044],{"align":919},"批量设置\u002F获取",[914,3046,3047,3051,3059],{},[933,3048,3049],{"align":919},[936,3050,951],{},[933,3052,3053,2985,3056],{"align":919},[1071,3054,3055],{},"HSET",[1071,3057,3058],{},"HGET",[933,3060,3061],{"align":919},"设置\u002F获取字段",[914,3063,3064,3066,3074],{},[933,3065],{"align":919},[933,3067,3068,2985,3071],{"align":919},[1071,3069,3070],{},"HMSET",[1071,3072,3073],{},"HMGET",[933,3075,3076],{"align":919},"批量操作字段",[914,3078,3079,3081,3086],{},[933,3080],{"align":919},[933,3082,3083],{"align":919},[1071,3084,3085],{},"HGETALL",[933,3087,3088],{"align":919},"获取所有字段和值",[914,3090,3091,3093,3098],{},[933,3092],{"align":919},[933,3094,3095],{"align":919},[1071,3096,3097],{},"HDEL",[933,3099,3100],{"align":919},"删除字段",[914,3102,3103,3107,3115],{},[933,3104,3105],{"align":919},[936,3106,964],{},[933,3108,3109,2985,3112],{"align":919},[1071,3110,3111],{},"LPUSH",[1071,3113,3114],{},"RPUSH",[933,3116,3117],{"align":919},"左推\u002F右推入队列",[914,3119,3120,3122,3130],{},[933,3121],{"align":919},[933,3123,3124,2985,3127],{"align":919},[1071,3125,3126],{},"LPOP",[1071,3128,3129],{},"RPOP",[933,3131,3132],{"align":919},"左弹\u002F右弹出队列",[914,3134,3135,3137,3142],{},[933,3136],{"align":919},[933,3138,3139],{"align":919},[1071,3140,3141],{},"LRANGE",[933,3143,3144],{"align":919},"获取指定范围元素（如 0 -1）",[914,3146,3147,3149,3157],{},[933,3148],{"align":919},[933,3150,3151,2985,3154],{"align":919},[1071,3152,3153],{},"BLPOP",[1071,3155,3156],{},"BRPOP",[933,3158,3159],{"align":919},"阻塞式弹出（消息队列常用）",[914,3161,3162,3166,3174],{},[933,3163,3164],{"align":919},[936,3165,977],{},[933,3167,3168,2985,3171],{"align":919},[1071,3169,3170],{},"SADD",[1071,3172,3173],{},"SREM",[933,3175,3176],{"align":919},"添加\u002F删除元素",[914,3178,3179,3181,3186],{},[933,3180],{"align":919},[933,3182,3183],{"align":919},[1071,3184,3185],{},"SMEMBERS",[933,3187,3188],{"align":919},"获取所有元素",[914,3190,3191,3193,3198],{},[933,3192],{"align":919},[933,3194,3195],{"align":919},[1071,3196,3197],{},"SISMEMBER",[933,3199,3200],{"align":919},"判断元素是否存在",[914,3202,3203,3205,3213],{},[933,3204],{"align":919},[933,3206,3207,2985,3210],{"align":919},[1071,3208,3209],{},"SINTER",[1071,3211,3212],{},"SUNION",[933,3214,3215],{"align":919},"交集 \u002F 并集",[914,3217,3218,3222,3227],{},[933,3219,3220],{"align":919},[936,3221,990],{},[933,3223,3224],{"align":919},[1071,3225,3226],{},"ZADD",[933,3228,3229],{"align":919},"添加元素（需指定 score）",[914,3231,3232,3234,3239],{},[933,3233],{"align":919},[933,3235,3236],{"align":919},[1071,3237,3238],{},"ZRANGE",[933,3240,3241],{"align":919},"按分数从小到大获取",[914,3243,3244,3246,3251],{},[933,3245],{"align":919},[933,3247,3248],{"align":919},[1071,3249,3250],{},"ZREVRANGE",[933,3252,3253],{"align":919},"按分数从大到小获取",[914,3255,3256,3258,3263],{},[933,3257],{"align":919},[933,3259,3260],{"align":919},[1071,3261,3262],{},"ZSCORE",[933,3264,3265],{"align":919},"获取元素分数",[914,3267,3268,3273,3277],{},[933,3269,3270],{"align":919},[936,3271,3272],{},"全局",[933,3274,3275],{"align":919},[1071,3276,2397],{},[933,3278,3279],{"align":919},"删除 Key",[914,3281,3282,3284,3292],{},[933,3283],{"align":919},[933,3285,3286,2985,3289],{"align":919},[1071,3287,3288],{},"EXPIRE",[1071,3290,3291],{},"TTL",[933,3293,3294],{"align":919},"设置过期时间 \u002F 查看剩余时间",[914,3296,3297,3299,3304],{},[933,3298],{"align":919},[933,3300,3301],{"align":919},[1071,3302,3303],{},"EXISTS",[933,3305,3306],{"align":919},"判断 Key 是否存在",[914,3308,3309,3311,3316],{},[933,3310],{"align":919},[933,3312,3313],{"align":919},[1071,3314,3315],{},"KEYS",[933,3317,3318],{"align":919},"查找所有符合模式的 Key（生产禁用）",[914,3320,3321,3323,3328],{},[933,3322],{"align":919},[933,3324,3325],{"align":919},[1071,3326,3327],{},"SCAN",[933,3329,3330],{"align":919},"渐进式遍历 Key（推荐）",[895,3332],{},[903,3334,3336],{"id":3335},"q26-keys-和-scan-的区别","Q26: KEYS 和 SCAN 的区别",[908,3338,3339,3351],{},[911,3340,3341],{},[914,3342,3343,3345,3348],{},[917,3344,1011],{"align":919},[917,3346,3347],{"align":919},"KEYS pattern",[917,3349,3350],{"align":919},"SCAN cursor",[928,3352,3353,3366,3379,3392],{},[914,3354,3355,3360,3363],{},[933,3356,3357],{"align":919},[936,3358,3359],{},"阻塞性",[933,3361,3362],{"align":919},"✅ 阻塞主线程",[933,3364,3365],{"align":919},"❌ 非阻塞，分批进行",[914,3367,3368,3373,3376],{},[933,3369,3370],{"align":919},[936,3371,3372],{},"执行效率",[933,3374,3375],{"align":919},"O(N) 一次性返回",[933,3377,3378],{"align":919},"O(1) 每次返回少量数据",[914,3380,3381,3386,3389],{},[933,3382,3383],{"align":919},[936,3384,3385],{},"生产环境",[933,3387,3388],{"align":919},"⚠️ 禁止使用",[933,3390,3391],{"align":919},"✅ 推荐使用",[914,3393,3394,3399,3402],{},[933,3395,3396],{"align":919},[936,3397,3398],{},"数据一致性",[933,3400,3401],{"align":919},"结果准确",[933,3403,3404],{"align":919},"可能会有重复或漏掉（期间有增删）",[895,3406],{},[898,3408,3410],{"id":3409},"八高频考点清单","八、高频考点清单",[903,3412,3413],{"id":3413},"必考",[1509,3415,3418,3427,3433,3439,3445,3451],{"className":3416},[3417],"contains-task-list",[1512,3419,3422,3426],{"className":3420},[3421],"task-list-item",[3423,3424],"input",{"disabled":1324,"type":3425},"checkbox"," 五种数据类型及底层实现",[1512,3428,3430,3432],{"className":3429},[3421],[3423,3431],{"disabled":1324,"type":3425}," 常见数据类型命令（SETNX, HGETALL, ZRANGE等）",[1512,3434,3436,3438],{"className":3435},[3421],[3423,3437],{"disabled":1324,"type":3425}," RDB vs AOF，同步策略",[1512,3440,3442,3444],{"className":3441},[3421],[3423,3443],{"disabled":1324,"type":3425}," 缓存穿透\u002F击穿\u002F雪崩解决方案",[1512,3446,3448,3450],{"className":3447},[3421],[3423,3449],{"disabled":1324,"type":3425}," 分布式锁正确实现",[1512,3452,3454,3456],{"className":3453},[3421],[3423,3455],{"disabled":1324,"type":3425}," 主从复制与哨兵选主流程",[903,3458,3459],{"id":3459},"常考",[1509,3461,3463,3469,3475,3481,3487,3493,3499],{"className":3462},[3417],[1512,3464,3466,3468],{"className":3465},[3421],[3423,3467],{"disabled":1324,"type":3425}," ZSet 为什么用跳表",[1512,3470,3472,3474],{"className":3471},[3421],[3423,3473],{"disabled":1324,"type":3425}," AOF 重写与混合持久化",[1512,3476,3478,3480],{"className":3477},[3421],[3423,3479],{"disabled":1324,"type":3425}," KEYS vs SCAN 的区别",[1512,3482,3484,3486],{"className":3483},[3421],[3423,3485],{"disabled":1324,"type":3425}," Cluster 槽位与 hash tag",[1512,3488,3490,3492],{"className":3489},[3421],[3423,3491],{"disabled":1324,"type":3425}," 内存淘汰策略",[1512,3494,3496,3498],{"className":3495},[3421],[3423,3497],{"disabled":1324,"type":3425}," 大 key 处理",[1512,3500,3502,3504],{"className":3501},[3421],[3423,3503],{"disabled":1324,"type":3425}," 过期删除策略",[903,3506,3507],{"id":3507},"进阶",[1509,3509,3511,3517,3523,3529,3535,3541],{"className":3510},[3417],[1512,3512,3514,3516],{"className":3513},[3421],[3423,3515],{"disabled":1324,"type":3425}," Redis 单线程为何高性能",[1512,3518,3520,3522],{"className":3519},[3421],[3423,3521],{"disabled":1324,"type":3425}," IO 多路复用原理",[1512,3524,3526,3528],{"className":3525},[3421],[3423,3527],{"disabled":1324,"type":3425}," Raft 选举（Sentinel\u002FCluster）",[1512,3530,3532,3534],{"className":3531},[3421],[3423,3533],{"disabled":1324,"type":3425}," 缓存与 DB 一致性方案",[1512,3536,3538,3540],{"className":3537},[3421],[3423,3539],{"disabled":1324,"type":3425}," Redlock 原理与争议",[1512,3542,3544,3546],{"className":3543},[3421],[3423,3545],{"disabled":1324,"type":3425}," 热点 Key 解决方案",[3548,3549,3550],"style",{},"html pre.shiki code .sCsY4, html code.shiki .sCsY4{--shiki-light:#6A737D;--shiki-default:#6A737D;--shiki-dark:#6A737D}html pre.shiki code .snPdu, html code.shiki .snPdu{--shiki-light:#6F42C1;--shiki-default:#6F42C1;--shiki-dark:#B392F0}html pre.shiki code .sIIMD, html code.shiki .sIIMD{--shiki-light:#032F62;--shiki-default:#032F62;--shiki-dark:#9ECBFF}html pre.shiki code .sBjJW, html code.shiki .sBjJW{--shiki-light:#005CC5;--shiki-default:#005CC5;--shiki-dark:#79B8FF}html .light .shiki span {color: var(--shiki-light);background: var(--shiki-light-bg);font-style: var(--shiki-light-font-style);font-weight: var(--shiki-light-font-weight);text-decoration: var(--shiki-light-text-decoration);}html.light .shiki span {color: var(--shiki-light);background: var(--shiki-light-bg);font-style: var(--shiki-light-font-style);font-weight: var(--shiki-light-font-weight);text-decoration: var(--shiki-light-text-decoration);}html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .dark .shiki span {color: var(--shiki-dark);background: var(--shiki-dark-bg);font-style: var(--shiki-dark-font-style);font-weight: var(--shiki-dark-font-weight);text-decoration: var(--shiki-dark-text-decoration);}html.dark .shiki span {color: var(--shiki-dark);background: var(--shiki-dark-bg);font-style: var(--shiki-dark-font-style);font-weight: var(--shiki-dark-font-weight);text-decoration: var(--shiki-dark-text-decoration);}html pre.shiki code .s8jYJ, html code.shiki .s8jYJ{--shiki-light:#D73A49;--shiki-default:#D73A49;--shiki-dark:#F97583}html pre.shiki code .sxrX7, html code.shiki .sxrX7{--shiki-light:#24292E;--shiki-default:#24292E;--shiki-dark:#E1E4E8}",{"title":11,"searchDepth":1088,"depth":1088,"links":3552},[3553,3559,3565,3570,3575,3581,3589,3593],{"id":900,"depth":1088,"text":901,"children":3554},[3555,3556,3557,3558],{"id":905,"depth":1101,"text":906},{"id":1001,"depth":1101,"text":1002},{"id":1138,"depth":1101,"text":1139},{"id":1210,"depth":1101,"text":1211},{"id":1362,"depth":1088,"text":1363,"children":3560},[3561,3562,3563,3564],{"id":1366,"depth":1101,"text":1367},{"id":1449,"depth":1101,"text":1450},{"id":1501,"depth":1101,"text":1502},{"id":1545,"depth":1101,"text":1546},{"id":1593,"depth":1088,"text":1594,"children":3566},[3567,3568,3569],{"id":1597,"depth":1101,"text":1598},{"id":1642,"depth":1101,"text":1643},{"id":1696,"depth":1101,"text":1697},{"id":1783,"depth":1088,"text":1784,"children":3571},[3572,3573,3574],{"id":1787,"depth":1101,"text":1788},{"id":2033,"depth":1101,"text":2034},{"id":2140,"depth":1101,"text":2141},{"id":2238,"depth":1088,"text":2239,"children":3576},[3577,3578,3579,3580],{"id":2242,"depth":1101,"text":2243},{"id":2329,"depth":1101,"text":2330},{"id":2411,"depth":1101,"text":2412},{"id":2481,"depth":1101,"text":2482},{"id":2553,"depth":1088,"text":2554,"children":3582},[3583,3584,3585,3586,3587,3588],{"id":2557,"depth":1101,"text":2558},{"id":2622,"depth":1101,"text":2623},{"id":2694,"depth":1101,"text":2695},{"id":2780,"depth":1101,"text":2781},{"id":2817,"depth":1101,"text":2818},{"id":2878,"depth":1101,"text":2879},{"id":2953,"depth":1088,"text":2954,"children":3590},[3591,3592],{"id":2957,"depth":1101,"text":2958},{"id":3335,"depth":1101,"text":3336},{"id":3409,"depth":1088,"text":3410,"children":3594},[3595,3596,3597],{"id":3413,"depth":1101,"text":3413},{"id":3459,"depth":1101,"text":3459},{"id":3507,"depth":1101,"text":3507},"md",{},{"title":350,"description":351},"interview\u002Fredis","UQI3SxlwtKbRR7inSx0YuYqxbfNvpyHmimzY7V1ZiIU",1775496433415]