[{"data":1,"prerenderedAt":3620},["ShallowReactive",2],{"search-docs":3,"doc-\u002Finterview\u002Frocketmq":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":354,"body":888,"description":355,"extension":3615,"meta":3616,"navigation":1149,"path":353,"seo":3617,"stem":3618,"__hash__":3619},"docs\u002Finterview\u002Frocketmq.md",{"type":889,"value":890,"toc":3570},"minimark",[891,894,897,902,907,969,974,996,998,1002,1013,1066,1068,1072,1114,1167,1169,1173,1177,1182,1232,1237,1368,1373,1410,1412,1416,1421,1429,1434,1510,1515,1529,1531,1535,1540,1601,1606,1792,1794,1798,1802,1807,1845,1850,2060,2062,2066,2071,2092,2097,2105,2107,2111,2115,2120,2158,2163,2200,2205,2216,2218,2222,2226,2232,2237,2456,2458,2462,2467,2478,2483,2499,2501,2505,2509,2562,2564,2568,2604,2634,2636,2640,2645,2656,2660,2716,2777,2779,2783,2787,2823,2869,2871,2875,2880,2891,2896,2946,2948,2952,2956,3009,3015,3017,3021,3075,3137,3139,3143,3149,3154,3165,3198,3200,3204,3296,3298,3302,3361,3367,3369,3373,3378,3390,3395,3409,3442,3444,3448,3451,3488,3491,3530,3533,3566],[892,893,355],"p",{},[895,896],"hr",{},[898,899,901],"h2",{"id":900},"一基础架构","一、基础架构",[903,904,906],"h3",{"id":905},"q1-rocketmq-的核心组件","Q1: RocketMQ 的核心组件",[908,909,910,924],"table",{},[911,912,913],"thead",{},[914,915,916,921],"tr",{},[917,918,920],"th",{"align":919},"left","组件",[917,922,923],{"align":919},"说明",[925,926,927,939,949,959],"tbody",{},[914,928,929,936],{},[930,931,932],"td",{"align":919},[933,934,935],"strong",{},"NameServer",[930,937,938],{"align":919},"注册中心，轻量级，无状态，多个互不通信",[914,940,941,946],{},[930,942,943],{"align":919},[933,944,945],{},"Broker",[930,947,948],{"align":919},"消息存储和转发，分 Master\u002FSlave",[914,950,951,956],{},[930,952,953],{"align":919},[933,954,955],{},"Producer",[930,957,958],{"align":919},"消息生产者",[914,960,961,966],{},[930,962,963],{"align":919},[933,964,965],{},"Consumer",[930,967,968],{"align":919},"消息消费者",[892,970,971],{},[933,972,973],{},"存储结构：",[975,976,977,984,990],"ul",{},[978,979,980,983],"li",{},[933,981,982],{},"CommitLog","：所有消息顺序追加写入",[978,985,986,989],{},[933,987,988],{},"ConsumeQueue","：消息消费队列，存储 offset 指向 CommitLog",[978,991,992,995],{},[933,993,994],{},"IndexFile","：消息索引，支持按 key\u002F时间查询",[895,997],{},[903,999,1001],{"id":1000},"q2-topicqueuetag-的关系","Q2: Topic、Queue、Tag 的关系",[1003,1004,1010],"pre",{"className":1005,"code":1007,"language":1008,"meta":1009},[1006],"language-text","flowchart LR\n    T[\"Topic #40;主题#41;\"] --> Q0[\"Queue 0\"] --> C1[\"Consumer Instance 1\"]\n    T --> Q1[\"Queue 1\"] --> C2[\"Consumer Instance 2\"]\n    T --> Q2[\"Queue 2\"] --> C3[\"Consumer Instance 3\"]\n    T --> Q3[\"Queue 3\"] --> C4[\"Consumer Instance 4\"]\n","text","mermaid",[1011,1012,1007],"code",{"__ignoreMap":11},[908,1014,1015,1024],{},[911,1016,1017],{},[914,1018,1019,1022],{},[917,1020,1021],{"align":919},"概念",[917,1023,923],{"align":919},[925,1025,1026,1036,1046,1056],{},[914,1027,1028,1033],{},[930,1029,1030],{"align":919},[933,1031,1032],{},"Topic",[930,1034,1035],{"align":919},"一类消息的集合，逻辑分类",[914,1037,1038,1043],{},[930,1039,1040],{"align":919},[933,1041,1042],{},"Queue",[930,1044,1045],{"align":919},"物理分区，一个 Topic 包含多个 Queue",[914,1047,1048,1053],{},[930,1049,1050],{"align":919},[933,1051,1052],{},"Tag",[930,1054,1055],{"align":919},"消息标签，二级分类，用于过滤",[914,1057,1058,1063],{},[930,1059,1060],{"align":919},[933,1061,1062],{},"Key",[930,1064,1065],{"align":919},"消息 Key，用于查询和去重",[895,1067],{},[903,1069,1071],{"id":1070},"q3-集群消费和广播消费的区别","Q3: 集群消费和广播消费的区别",[908,1073,1074,1086],{},[911,1075,1076],{},[914,1077,1078,1081,1083],{},[917,1079,1080],{"align":919},"模式",[917,1082,923],{"align":919},[917,1084,1085],{"align":919},"场景",[925,1087,1088,1101],{},[914,1089,1090,1095,1098],{},[930,1091,1092],{"align":919},[933,1093,1094],{},"集群消费",[930,1096,1097],{"align":919},"一条消息只被消费组内一个实例消费",[930,1099,1100],{"align":919},"业务处理（默认）",[914,1102,1103,1108,1111],{},[930,1104,1105],{"align":919},[933,1106,1107],{},"广播消费",[930,1109,1110],{"align":919},"一条消息被消费组内所有实例消费",[930,1112,1113],{"align":919},"本地缓存刷新",[1003,1115,1119],{"className":1116,"code":1117,"language":1118,"meta":11,"style":11},"language-java shiki shiki-themes github-light github-light github-dark","\u002F\u002F 集群消费（默认）\nconsumer.setMessageModel(MessageModel.CLUSTERING);\n\n\u002F\u002F 广播消费\nconsumer.setMessageModel(MessageModel.BROADCASTING);\n","java",[1011,1120,1121,1130,1144,1151,1157],{"__ignoreMap":11},[1122,1123,1126],"span",{"class":1124,"line":1125},"line",1,[1122,1127,1129],{"class":1128},"sCsY4","\u002F\u002F 集群消费（默认）\n",[1122,1131,1133,1137,1141],{"class":1124,"line":1132},2,[1122,1134,1136],{"class":1135},"sxrX7","consumer.",[1122,1138,1140],{"class":1139},"snPdu","setMessageModel",[1122,1142,1143],{"class":1135},"(MessageModel.CLUSTERING);\n",[1122,1145,1147],{"class":1124,"line":1146},3,[1122,1148,1150],{"emptyLinePlaceholder":1149},true,"\n",[1122,1152,1154],{"class":1124,"line":1153},4,[1122,1155,1156],{"class":1128},"\u002F\u002F 广播消费\n",[1122,1158,1160,1162,1164],{"class":1124,"line":1159},5,[1122,1161,1136],{"class":1135},[1122,1163,1140],{"class":1139},[1122,1165,1166],{"class":1135},"(MessageModel.BROADCASTING);\n",[895,1168],{},[898,1170,1172],{"id":1171},"二消息可靠性","二、消息可靠性",[903,1174,1176],{"id":1175},"q4-如何保证消息不丢失","Q4: 如何保证消息不丢失",[892,1178,1179],{},[933,1180,1181],{},"消息丢失的三个环节：",[908,1183,1184,1197],{},[911,1185,1186],{},[914,1187,1188,1191,1194],{},[917,1189,1190],{"align":919},"环节",[917,1192,1193],{"align":919},"风险点",[917,1195,1196],{"align":919},"解决方案",[925,1198,1199,1210,1221],{},[914,1200,1201,1204,1207],{},[930,1202,1203],{"align":919},"生产者→Broker",[930,1205,1206],{"align":919},"网络异常",[930,1208,1209],{"align":919},"同步发送 + 重试",[914,1211,1212,1215,1218],{},[930,1213,1214],{"align":919},"Broker 存储",[930,1216,1217],{"align":919},"宕机丢失",[930,1219,1220],{"align":919},"同步刷盘 + 主从同步",[914,1222,1223,1226,1229],{},[930,1224,1225],{"align":919},"Broker→消费者",[930,1227,1228],{"align":919},"消费失败",[930,1230,1231],{"align":919},"手动 ACK + 重试",[892,1233,1234],{},[933,1235,1236],{},"生产者可靠发送：",[1003,1238,1240],{"className":1116,"code":1239,"language":1118,"meta":11,"style":11},"\u002F\u002F 同步发送（推荐）\nSendResult result = producer.send(msg);\n\n\u002F\u002F 异步发送 + 回调\nproducer.send(msg, new SendCallback() {\n    @Override\n    public void onSuccess(SendResult sendResult) {}\n    @Override\n    public void onException(Throwable e) {\n        \u002F\u002F 重试或记录日志\n    }\n});\n",[1011,1241,1242,1247,1265,1269,1274,1293,1302,1324,1331,1350,1356,1362],{"__ignoreMap":11},[1122,1243,1244],{"class":1124,"line":1125},[1122,1245,1246],{"class":1128},"\u002F\u002F 同步发送（推荐）\n",[1122,1248,1249,1252,1256,1259,1262],{"class":1124,"line":1132},[1122,1250,1251],{"class":1135},"SendResult result ",[1122,1253,1255],{"class":1254},"s8jYJ","=",[1122,1257,1258],{"class":1135}," producer.",[1122,1260,1261],{"class":1139},"send",[1122,1263,1264],{"class":1135},"(msg);\n",[1122,1266,1267],{"class":1124,"line":1146},[1122,1268,1150],{"emptyLinePlaceholder":1149},[1122,1270,1271],{"class":1124,"line":1153},[1122,1272,1273],{"class":1128},"\u002F\u002F 异步发送 + 回调\n",[1122,1275,1276,1279,1281,1284,1287,1290],{"class":1124,"line":1159},[1122,1277,1278],{"class":1135},"producer.",[1122,1280,1261],{"class":1139},[1122,1282,1283],{"class":1135},"(msg, ",[1122,1285,1286],{"class":1254},"new",[1122,1288,1289],{"class":1139}," SendCallback",[1122,1291,1292],{"class":1135},"() {\n",[1122,1294,1296,1299],{"class":1124,"line":1295},6,[1122,1297,1298],{"class":1135},"    @",[1122,1300,1301],{"class":1254},"Override\n",[1122,1303,1305,1308,1311,1314,1317,1321],{"class":1124,"line":1304},7,[1122,1306,1307],{"class":1254},"    public",[1122,1309,1310],{"class":1254}," void",[1122,1312,1313],{"class":1139}," onSuccess",[1122,1315,1316],{"class":1135},"(SendResult ",[1122,1318,1320],{"class":1319},"sP4rz","sendResult",[1122,1322,1323],{"class":1135},") {}\n",[1122,1325,1327,1329],{"class":1124,"line":1326},8,[1122,1328,1298],{"class":1135},[1122,1330,1301],{"class":1254},[1122,1332,1334,1336,1338,1341,1344,1347],{"class":1124,"line":1333},9,[1122,1335,1307],{"class":1254},[1122,1337,1310],{"class":1254},[1122,1339,1340],{"class":1139}," onException",[1122,1342,1343],{"class":1135},"(Throwable ",[1122,1345,1346],{"class":1319},"e",[1122,1348,1349],{"class":1135},") {\n",[1122,1351,1353],{"class":1124,"line":1352},10,[1122,1354,1355],{"class":1128},"        \u002F\u002F 重试或记录日志\n",[1122,1357,1359],{"class":1124,"line":1358},11,[1122,1360,1361],{"class":1135},"    }\n",[1122,1363,1365],{"class":1124,"line":1364},12,[1122,1366,1367],{"class":1135},"});\n",[892,1369,1370],{},[933,1371,1372],{},"Broker 配置：",[1003,1374,1378],{"className":1375,"code":1376,"language":1377,"meta":11,"style":11},"language-properties shiki shiki-themes github-light github-light github-dark","# 同步刷盘（更可靠）\nflushDiskType=SYNC_FLUSH\n\n# 同步复制（更可靠）\nbrokerRole=SYNC_MASTER\n","properties",[1011,1379,1380,1385,1393,1397,1402],{"__ignoreMap":11},[1122,1381,1382],{"class":1124,"line":1125},[1122,1383,1384],{"class":1128},"# 同步刷盘（更可靠）\n",[1122,1386,1387,1390],{"class":1124,"line":1132},[1122,1388,1389],{"class":1254},"flushDiskType",[1122,1391,1392],{"class":1135},"=SYNC_FLUSH\n",[1122,1394,1395],{"class":1124,"line":1146},[1122,1396,1150],{"emptyLinePlaceholder":1149},[1122,1398,1399],{"class":1124,"line":1153},[1122,1400,1401],{"class":1128},"# 同步复制（更可靠）\n",[1122,1403,1404,1407],{"class":1124,"line":1159},[1122,1405,1406],{"class":1254},"brokerRole",[1122,1408,1409],{"class":1135},"=SYNC_MASTER\n",[895,1411],{},[903,1413,1415],{"id":1414},"q5-消费重试机制","Q5: 消费重试机制",[892,1417,1418],{},[933,1419,1420],{},"重试次数：",[975,1422,1423,1426],{},[978,1424,1425],{},"默认最多重试 16 次",[978,1427,1428],{},"重试间隔逐渐增大（10s、30s、1m、2m...）",[892,1430,1431],{},[933,1432,1433],{},"重试触发：",[1003,1435,1437],{"className":1116,"code":1436,"language":1118,"meta":11,"style":11},"consumer.registerMessageListener((msgs, context) -> {\n    try {\n        \u002F\u002F 处理消息\n        return ConsumeConcurrentlyStatus.CONSUME_SUCCESS;\n    } catch (Exception e) {\n        \u002F\u002F 返回 RECONSUME_LATER 触发重试\n        return ConsumeConcurrentlyStatus.RECONSUME_LATER;\n    }\n});\n",[1011,1438,1439,1455,1462,1467,1475,1490,1495,1502,1506],{"__ignoreMap":11},[1122,1440,1441,1443,1446,1449,1452],{"class":1124,"line":1125},[1122,1442,1136],{"class":1135},[1122,1444,1445],{"class":1139},"registerMessageListener",[1122,1447,1448],{"class":1135},"((msgs, context) ",[1122,1450,1451],{"class":1254},"->",[1122,1453,1454],{"class":1135}," {\n",[1122,1456,1457,1460],{"class":1124,"line":1132},[1122,1458,1459],{"class":1254},"    try",[1122,1461,1454],{"class":1135},[1122,1463,1464],{"class":1124,"line":1146},[1122,1465,1466],{"class":1128},"        \u002F\u002F 处理消息\n",[1122,1468,1469,1472],{"class":1124,"line":1153},[1122,1470,1471],{"class":1254},"        return",[1122,1473,1474],{"class":1135}," ConsumeConcurrentlyStatus.CONSUME_SUCCESS;\n",[1122,1476,1477,1480,1483,1486,1488],{"class":1124,"line":1159},[1122,1478,1479],{"class":1135},"    } ",[1122,1481,1482],{"class":1254},"catch",[1122,1484,1485],{"class":1135}," (Exception ",[1122,1487,1346],{"class":1319},[1122,1489,1349],{"class":1135},[1122,1491,1492],{"class":1124,"line":1295},[1122,1493,1494],{"class":1128},"        \u002F\u002F 返回 RECONSUME_LATER 触发重试\n",[1122,1496,1497,1499],{"class":1124,"line":1304},[1122,1498,1471],{"class":1254},[1122,1500,1501],{"class":1135}," ConsumeConcurrentlyStatus.RECONSUME_LATER;\n",[1122,1503,1504],{"class":1124,"line":1326},[1122,1505,1361],{"class":1135},[1122,1507,1508],{"class":1124,"line":1333},[1122,1509,1367],{"class":1135},[892,1511,1512],{},[933,1513,1514],{},"死信队列（DLQ）：",[975,1516,1517,1520,1526],{},[978,1518,1519],{},"重试超过最大次数后进入死信队列",[978,1521,1522,1523],{},"队列名：",[1011,1524,1525],{},"%DLQ%消费组名",[978,1527,1528],{},"需要手动处理或设置监控",[895,1530],{},[903,1532,1534],{"id":1533},"q6-如何保证幂等性","Q6: 如何保证幂等性",[892,1536,1537],{},[933,1538,1539],{},"常见方案：",[908,1541,1542,1555],{},[911,1543,1544],{},[914,1545,1546,1549,1552],{},[917,1547,1548],{"align":919},"方案",[917,1550,1551],{"align":919},"实现",[917,1553,1554],{"align":919},"适用场景",[925,1556,1557,1568,1579,1590],{},[914,1558,1559,1562,1565],{},[930,1560,1561],{"align":919},"唯一键",[930,1563,1564],{"align":919},"DB 唯一索引",[930,1566,1567],{"align":919},"写入操作",[914,1569,1570,1573,1576],{},[930,1571,1572],{"align":919},"去重表",[930,1574,1575],{"align":919},"消息 ID 存 Redis\u002FDB",[930,1577,1578],{"align":919},"通用",[914,1580,1581,1584,1587],{},[930,1582,1583],{"align":919},"状态机",[930,1585,1586],{"align":919},"检查状态再处理",[930,1588,1589],{"align":919},"状态流转",[914,1591,1592,1595,1598],{},[930,1593,1594],{"align":919},"Token",[930,1596,1597],{"align":919},"一次性 Token",[930,1599,1600],{"align":919},"防重提交",[892,1602,1603],{},[933,1604,1605],{},"去重表实现：",[1003,1607,1609],{"className":1116,"code":1608,"language":1118,"meta":11,"style":11},"public boolean consume(Message msg) {\n    String msgId = msg.getMsgId();\n    \n    \u002F\u002F 分布式锁 + 去重\n    if (redis.setNx(\"consumed:\" + msgId, \"1\", 7*24*3600)) {\n        try {\n            process(msg);\n            return true;\n        } catch (Exception e) {\n            redis.del(\"consumed:\" + msgId);\n            throw e;\n        }\n    }\n    return true;  \u002F\u002F 已消费，直接返回成功\n}\n",[1011,1610,1611,1625,1641,1646,1651,1699,1706,1713,1724,1737,1754,1762,1767,1772,1786],{"__ignoreMap":11},[1122,1612,1613,1616,1619,1622],{"class":1124,"line":1125},[1122,1614,1615],{"class":1254},"public",[1122,1617,1618],{"class":1254}," boolean",[1122,1620,1621],{"class":1139}," consume",[1122,1623,1624],{"class":1135},"(Message msg) {\n",[1122,1626,1627,1630,1632,1635,1638],{"class":1124,"line":1132},[1122,1628,1629],{"class":1135},"    String msgId ",[1122,1631,1255],{"class":1254},[1122,1633,1634],{"class":1135}," msg.",[1122,1636,1637],{"class":1139},"getMsgId",[1122,1639,1640],{"class":1135},"();\n",[1122,1642,1643],{"class":1124,"line":1146},[1122,1644,1645],{"class":1135},"    \n",[1122,1647,1648],{"class":1124,"line":1153},[1122,1649,1650],{"class":1128},"    \u002F\u002F 分布式锁 + 去重\n",[1122,1652,1653,1656,1659,1662,1665,1669,1672,1675,1678,1681,1685,1688,1691,1693,1696],{"class":1124,"line":1159},[1122,1654,1655],{"class":1254},"    if",[1122,1657,1658],{"class":1135}," (redis.",[1122,1660,1661],{"class":1139},"setNx",[1122,1663,1664],{"class":1135},"(",[1122,1666,1668],{"class":1667},"sIIMD","\"consumed:\"",[1122,1670,1671],{"class":1254}," +",[1122,1673,1674],{"class":1135}," msgId, ",[1122,1676,1677],{"class":1667},"\"1\"",[1122,1679,1680],{"class":1135},", ",[1122,1682,1684],{"class":1683},"sBjJW","7",[1122,1686,1687],{"class":1254},"*",[1122,1689,1690],{"class":1683},"24",[1122,1692,1687],{"class":1254},[1122,1694,1695],{"class":1683},"3600",[1122,1697,1698],{"class":1135},")) {\n",[1122,1700,1701,1704],{"class":1124,"line":1295},[1122,1702,1703],{"class":1254},"        try",[1122,1705,1454],{"class":1135},[1122,1707,1708,1711],{"class":1124,"line":1304},[1122,1709,1710],{"class":1139},"            process",[1122,1712,1264],{"class":1135},[1122,1714,1715,1718,1721],{"class":1124,"line":1326},[1122,1716,1717],{"class":1254},"            return",[1122,1719,1720],{"class":1683}," true",[1122,1722,1723],{"class":1135},";\n",[1122,1725,1726,1729,1731,1733,1735],{"class":1124,"line":1333},[1122,1727,1728],{"class":1135},"        } ",[1122,1730,1482],{"class":1254},[1122,1732,1485],{"class":1135},[1122,1734,1346],{"class":1319},[1122,1736,1349],{"class":1135},[1122,1738,1739,1742,1745,1747,1749,1751],{"class":1124,"line":1352},[1122,1740,1741],{"class":1135},"            redis.",[1122,1743,1744],{"class":1139},"del",[1122,1746,1664],{"class":1135},[1122,1748,1668],{"class":1667},[1122,1750,1671],{"class":1254},[1122,1752,1753],{"class":1135}," msgId);\n",[1122,1755,1756,1759],{"class":1124,"line":1358},[1122,1757,1758],{"class":1254},"            throw",[1122,1760,1761],{"class":1135}," e;\n",[1122,1763,1764],{"class":1124,"line":1364},[1122,1765,1766],{"class":1135},"        }\n",[1122,1768,1770],{"class":1124,"line":1769},13,[1122,1771,1361],{"class":1135},[1122,1773,1775,1778,1780,1783],{"class":1124,"line":1774},14,[1122,1776,1777],{"class":1254},"    return",[1122,1779,1720],{"class":1683},[1122,1781,1782],{"class":1135},";  ",[1122,1784,1785],{"class":1128},"\u002F\u002F 已消费，直接返回成功\n",[1122,1787,1789],{"class":1124,"line":1788},15,[1122,1790,1791],{"class":1135},"}\n",[895,1793],{},[898,1795,1797],{"id":1796},"三顺序消息","三、顺序消息",[903,1799,1801],{"id":1800},"q7-如何保证消息顺序","Q7: 如何保证消息顺序",[892,1803,1804],{},[933,1805,1806],{},"全局顺序 vs 分区顺序：",[908,1808,1809,1821],{},[911,1810,1811],{},[914,1812,1813,1816,1818],{},[917,1814,1815],{"align":919},"类型",[917,1817,923],{"align":919},[917,1819,1820],{"align":919},"性能",[925,1822,1823,1834],{},[914,1824,1825,1828,1831],{},[930,1826,1827],{"align":919},"全局顺序",[930,1829,1830],{"align":919},"整个 Topic 只用一个 Queue",[930,1832,1833],{"align":919},"差",[914,1835,1836,1839,1842],{},[930,1837,1838],{"align":919},"分区顺序",[930,1840,1841],{"align":919},"同一业务 key 进同一 Queue",[930,1843,1844],{"align":919},"好（推荐）",[892,1846,1847],{},[933,1848,1849],{},"分区顺序实现：",[1003,1851,1853],{"className":1116,"code":1852,"language":1118,"meta":11,"style":11},"\u002F\u002F 发送端：相同订单号进同一队列\nproducer.send(msg, new MessageQueueSelector() {\n    @Override\n    public MessageQueue select(List\u003CMessageQueue> mqs, Message msg, Object arg) {\n        Long orderId = (Long) arg;\n        int index = (int) (orderId % mqs.size());\n        return mqs.get(index);\n    }\n}, orderId);\n\n\u002F\u002F 消费端：使用顺序消费监听器\nconsumer.registerMessageListener(new MessageListenerOrderly() {\n    @Override\n    public ConsumeOrderlyStatus consumeMessage(List\u003CMessageExt> msgs, ConsumeOrderlyContext context) {\n        \u002F\u002F 顺序处理\n        return ConsumeOrderlyStatus.SUCCESS;\n    }\n});\n",[1011,1854,1855,1860,1875,1881,1917,1927,1958,1970,1974,1979,1983,1988,2003,2009,2037,2042,2050,2055],{"__ignoreMap":11},[1122,1856,1857],{"class":1124,"line":1125},[1122,1858,1859],{"class":1128},"\u002F\u002F 发送端：相同订单号进同一队列\n",[1122,1861,1862,1864,1866,1868,1870,1873],{"class":1124,"line":1132},[1122,1863,1278],{"class":1135},[1122,1865,1261],{"class":1139},[1122,1867,1283],{"class":1135},[1122,1869,1286],{"class":1254},[1122,1871,1872],{"class":1139}," MessageQueueSelector",[1122,1874,1292],{"class":1135},[1122,1876,1877,1879],{"class":1124,"line":1146},[1122,1878,1298],{"class":1135},[1122,1880,1301],{"class":1254},[1122,1882,1883,1885,1888,1891,1894,1897,1900,1903,1906,1909,1912,1915],{"class":1124,"line":1153},[1122,1884,1307],{"class":1254},[1122,1886,1887],{"class":1135}," MessageQueue ",[1122,1889,1890],{"class":1139},"select",[1122,1892,1893],{"class":1135},"(List\u003C",[1122,1895,1896],{"class":1254},"MessageQueue",[1122,1898,1899],{"class":1135},"> ",[1122,1901,1902],{"class":1319},"mqs",[1122,1904,1905],{"class":1135},", Message ",[1122,1907,1908],{"class":1319},"msg",[1122,1910,1911],{"class":1135},", Object ",[1122,1913,1914],{"class":1319},"arg",[1122,1916,1349],{"class":1135},[1122,1918,1919,1922,1924],{"class":1124,"line":1159},[1122,1920,1921],{"class":1135},"        Long orderId ",[1122,1923,1255],{"class":1254},[1122,1925,1926],{"class":1135}," (Long) arg;\n",[1122,1928,1929,1932,1935,1937,1940,1943,1946,1949,1952,1955],{"class":1124,"line":1295},[1122,1930,1931],{"class":1254},"        int",[1122,1933,1934],{"class":1135}," index ",[1122,1936,1255],{"class":1254},[1122,1938,1939],{"class":1135}," (",[1122,1941,1942],{"class":1254},"int",[1122,1944,1945],{"class":1135},") (orderId ",[1122,1947,1948],{"class":1254},"%",[1122,1950,1951],{"class":1135}," mqs.",[1122,1953,1954],{"class":1139},"size",[1122,1956,1957],{"class":1135},"());\n",[1122,1959,1960,1962,1964,1967],{"class":1124,"line":1304},[1122,1961,1471],{"class":1254},[1122,1963,1951],{"class":1135},[1122,1965,1966],{"class":1139},"get",[1122,1968,1969],{"class":1135},"(index);\n",[1122,1971,1972],{"class":1124,"line":1326},[1122,1973,1361],{"class":1135},[1122,1975,1976],{"class":1124,"line":1333},[1122,1977,1978],{"class":1135},"}, orderId);\n",[1122,1980,1981],{"class":1124,"line":1352},[1122,1982,1150],{"emptyLinePlaceholder":1149},[1122,1984,1985],{"class":1124,"line":1358},[1122,1986,1987],{"class":1128},"\u002F\u002F 消费端：使用顺序消费监听器\n",[1122,1989,1990,1992,1994,1996,1998,2001],{"class":1124,"line":1364},[1122,1991,1136],{"class":1135},[1122,1993,1445],{"class":1139},[1122,1995,1664],{"class":1135},[1122,1997,1286],{"class":1254},[1122,1999,2000],{"class":1139}," MessageListenerOrderly",[1122,2002,1292],{"class":1135},[1122,2004,2005,2007],{"class":1124,"line":1769},[1122,2006,1298],{"class":1135},[1122,2008,1301],{"class":1254},[1122,2010,2011,2013,2016,2019,2021,2024,2026,2029,2032,2035],{"class":1124,"line":1774},[1122,2012,1307],{"class":1254},[1122,2014,2015],{"class":1135}," ConsumeOrderlyStatus ",[1122,2017,2018],{"class":1139},"consumeMessage",[1122,2020,1893],{"class":1135},[1122,2022,2023],{"class":1254},"MessageExt",[1122,2025,1899],{"class":1135},[1122,2027,2028],{"class":1319},"msgs",[1122,2030,2031],{"class":1135},", ConsumeOrderlyContext ",[1122,2033,2034],{"class":1319},"context",[1122,2036,1349],{"class":1135},[1122,2038,2039],{"class":1124,"line":1788},[1122,2040,2041],{"class":1128},"        \u002F\u002F 顺序处理\n",[1122,2043,2045,2047],{"class":1124,"line":2044},16,[1122,2046,1471],{"class":1254},[1122,2048,2049],{"class":1135}," ConsumeOrderlyStatus.SUCCESS;\n",[1122,2051,2053],{"class":1124,"line":2052},17,[1122,2054,1361],{"class":1135},[1122,2056,2058],{"class":1124,"line":2057},18,[1122,2059,1367],{"class":1135},[895,2061],{},[903,2063,2065],{"id":2064},"q8-顺序消息的局限性","Q8: 顺序消息的局限性",[892,2067,2068],{},[933,2069,2070],{},"问题场景：",[2072,2073,2074,2080,2086],"ol",{},[978,2075,2076,2079],{},[933,2077,2078],{},"Broker 宕机","：队列迁移可能乱序",[978,2081,2082,2085],{},[933,2083,2084],{},"消费超时","：队列锁转移给其他消费者",[978,2087,2088,2091],{},[933,2089,2090],{},"扩缩容","：Queue 变化导致路由改变",[892,2093,2094],{},[933,2095,2096],{},"解决方案：",[975,2098,2099,2102],{},[978,2100,2101],{},"消费端检查顺序（如检查状态流转是否合法）",[978,2103,2104],{},"异常情况下降级处理",[895,2106],{},[898,2108,2110],{"id":2109},"四延迟消息","四、延迟消息",[903,2112,2114],{"id":2113},"q9-延迟消息的实现","Q9: 延迟消息的实现",[892,2116,2117],{},[933,2118,2119],{},"固定延迟级别（RocketMQ 4.x）：",[1003,2121,2123],{"className":1116,"code":2122,"language":1118,"meta":11,"style":11},"msg.setDelayTimeLevel(3);  \u002F\u002F 10s 后投递\n\n\u002F\u002F 延迟级别（18 个）:\n\u002F\u002F 1s 5s 10s 30s 1m 2m 3m 4m 5m 6m 7m 8m 9m 10m 20m 30m 1h 2h\n",[1011,2124,2125,2144,2148,2153],{"__ignoreMap":11},[1122,2126,2127,2130,2133,2135,2138,2141],{"class":1124,"line":1125},[1122,2128,2129],{"class":1135},"msg.",[1122,2131,2132],{"class":1139},"setDelayTimeLevel",[1122,2134,1664],{"class":1135},[1122,2136,2137],{"class":1683},"3",[1122,2139,2140],{"class":1135},");  ",[1122,2142,2143],{"class":1128},"\u002F\u002F 10s 后投递\n",[1122,2145,2146],{"class":1124,"line":1132},[1122,2147,1150],{"emptyLinePlaceholder":1149},[1122,2149,2150],{"class":1124,"line":1146},[1122,2151,2152],{"class":1128},"\u002F\u002F 延迟级别（18 个）:\n",[1122,2154,2155],{"class":1124,"line":1153},[1122,2156,2157],{"class":1128},"\u002F\u002F 1s 5s 10s 30s 1m 2m 3m 4m 5m 6m 7m 8m 9m 10m 20m 30m 1h 2h\n",[892,2159,2160],{},[933,2161,2162],{},"任意时间延迟（RocketMQ 5.0+）：",[1003,2164,2166],{"className":1116,"code":2165,"language":1118,"meta":11,"style":11},"\u002F\u002F 设置投递时间戳\nmsg.setDeliverTimeMs(System.currentTimeMillis() + 60000);  \u002F\u002F 1 分钟后\n",[1011,2167,2168,2173],{"__ignoreMap":11},[1122,2169,2170],{"class":1124,"line":1125},[1122,2171,2172],{"class":1128},"\u002F\u002F 设置投递时间戳\n",[1122,2174,2175,2177,2180,2183,2186,2189,2192,2195,2197],{"class":1124,"line":1132},[1122,2176,2129],{"class":1135},[1122,2178,2179],{"class":1139},"setDeliverTimeMs",[1122,2181,2182],{"class":1135},"(System.",[1122,2184,2185],{"class":1139},"currentTimeMillis",[1122,2187,2188],{"class":1135},"() ",[1122,2190,2191],{"class":1254},"+",[1122,2193,2194],{"class":1683}," 60000",[1122,2196,2140],{"class":1135},[1122,2198,2199],{"class":1128},"\u002F\u002F 1 分钟后\n",[892,2201,2202],{},[933,2203,2204],{},"原理：",[975,2206,2207,2210,2213],{},[978,2208,2209],{},"延迟消息先存入特殊 Topic（SCHEDULE_TOPIC_XXXX）",[978,2211,2212],{},"定时任务检查到期消息",[978,2214,2215],{},"到期后投递到真正的 Topic",[895,2217],{},[898,2219,2221],{"id":2220},"五事务消息","五、事务消息",[903,2223,2225],{"id":2224},"q10-事务消息的实现流程","Q10: 事务消息的实现流程",[1003,2227,2230],{"className":2228,"code":2229,"language":1008,"meta":1009},[1006],"sequenceDiagram\n    participant P as Producer\n    participant B as Broker\n    participant C as Consumer\n    P->>B: ① 发送半消息#40;Prepare#41;\n    B-->>P: ② 返回发送结果\n    P->>P: ③ 执行本地事务\n    P->>B: ④ Commit\u002FRollback\n    B->>C: ⑤ 消息可消费\n    B-->>P: ⑥ 回查本地事务状态#40;如需#41;\n",[1011,2231,2229],{"__ignoreMap":11},[892,2233,2234],{},[933,2235,2236],{},"代码示例：",[1003,2238,2240],{"className":1116,"code":2239,"language":1118,"meta":11,"style":11},"TransactionMQProducer producer = new TransactionMQProducer(\"group\");\nproducer.setTransactionListener(new TransactionListener() {\n    @Override\n    public LocalTransactionState executeLocalTransaction(Message msg, Object arg) {\n        try {\n            \u002F\u002F 执行本地事务\n            doLocalTransaction();\n            return LocalTransactionState.COMMIT_MESSAGE;\n        } catch (Exception e) {\n            return LocalTransactionState.ROLLBACK_MESSAGE;\n        }\n    }\n    \n    @Override\n    public LocalTransactionState checkLocalTransaction(MessageExt msg) {\n        \u002F\u002F 回查本地事务状态\n        if (checkTransactionSuccess(msg)) {\n            return LocalTransactionState.COMMIT_MESSAGE;\n        }\n        return LocalTransactionState.ROLLBACK_MESSAGE;\n    }\n});\n\n\u002F\u002F 发送事务消息\nproducer.sendMessageInTransaction(msg, null);\n",[1011,2241,2242,2263,2279,2285,2306,2312,2317,2324,2331,2343,2350,2354,2358,2362,2368,2384,2389,2402,2408,2413,2420,2425,2430,2435,2441],{"__ignoreMap":11},[1122,2243,2244,2247,2249,2252,2255,2257,2260],{"class":1124,"line":1125},[1122,2245,2246],{"class":1135},"TransactionMQProducer producer ",[1122,2248,1255],{"class":1254},[1122,2250,2251],{"class":1254}," new",[1122,2253,2254],{"class":1139}," TransactionMQProducer",[1122,2256,1664],{"class":1135},[1122,2258,2259],{"class":1667},"\"group\"",[1122,2261,2262],{"class":1135},");\n",[1122,2264,2265,2267,2270,2272,2274,2277],{"class":1124,"line":1132},[1122,2266,1278],{"class":1135},[1122,2268,2269],{"class":1139},"setTransactionListener",[1122,2271,1664],{"class":1135},[1122,2273,1286],{"class":1254},[1122,2275,2276],{"class":1139}," TransactionListener",[1122,2278,1292],{"class":1135},[1122,2280,2281,2283],{"class":1124,"line":1146},[1122,2282,1298],{"class":1135},[1122,2284,1301],{"class":1254},[1122,2286,2287,2289,2292,2295,2298,2300,2302,2304],{"class":1124,"line":1153},[1122,2288,1307],{"class":1254},[1122,2290,2291],{"class":1135}," LocalTransactionState ",[1122,2293,2294],{"class":1139},"executeLocalTransaction",[1122,2296,2297],{"class":1135},"(Message ",[1122,2299,1908],{"class":1319},[1122,2301,1911],{"class":1135},[1122,2303,1914],{"class":1319},[1122,2305,1349],{"class":1135},[1122,2307,2308,2310],{"class":1124,"line":1159},[1122,2309,1703],{"class":1254},[1122,2311,1454],{"class":1135},[1122,2313,2314],{"class":1124,"line":1295},[1122,2315,2316],{"class":1128},"            \u002F\u002F 执行本地事务\n",[1122,2318,2319,2322],{"class":1124,"line":1304},[1122,2320,2321],{"class":1139},"            doLocalTransaction",[1122,2323,1640],{"class":1135},[1122,2325,2326,2328],{"class":1124,"line":1326},[1122,2327,1717],{"class":1254},[1122,2329,2330],{"class":1135}," LocalTransactionState.COMMIT_MESSAGE;\n",[1122,2332,2333,2335,2337,2339,2341],{"class":1124,"line":1333},[1122,2334,1728],{"class":1135},[1122,2336,1482],{"class":1254},[1122,2338,1485],{"class":1135},[1122,2340,1346],{"class":1319},[1122,2342,1349],{"class":1135},[1122,2344,2345,2347],{"class":1124,"line":1352},[1122,2346,1717],{"class":1254},[1122,2348,2349],{"class":1135}," LocalTransactionState.ROLLBACK_MESSAGE;\n",[1122,2351,2352],{"class":1124,"line":1358},[1122,2353,1766],{"class":1135},[1122,2355,2356],{"class":1124,"line":1364},[1122,2357,1361],{"class":1135},[1122,2359,2360],{"class":1124,"line":1769},[1122,2361,1645],{"class":1135},[1122,2363,2364,2366],{"class":1124,"line":1774},[1122,2365,1298],{"class":1135},[1122,2367,1301],{"class":1254},[1122,2369,2370,2372,2374,2377,2380,2382],{"class":1124,"line":1788},[1122,2371,1307],{"class":1254},[1122,2373,2291],{"class":1135},[1122,2375,2376],{"class":1139},"checkLocalTransaction",[1122,2378,2379],{"class":1135},"(MessageExt ",[1122,2381,1908],{"class":1319},[1122,2383,1349],{"class":1135},[1122,2385,2386],{"class":1124,"line":2044},[1122,2387,2388],{"class":1128},"        \u002F\u002F 回查本地事务状态\n",[1122,2390,2391,2394,2396,2399],{"class":1124,"line":2052},[1122,2392,2393],{"class":1254},"        if",[1122,2395,1939],{"class":1135},[1122,2397,2398],{"class":1139},"checkTransactionSuccess",[1122,2400,2401],{"class":1135},"(msg)) {\n",[1122,2403,2404,2406],{"class":1124,"line":2057},[1122,2405,1717],{"class":1254},[1122,2407,2330],{"class":1135},[1122,2409,2411],{"class":1124,"line":2410},19,[1122,2412,1766],{"class":1135},[1122,2414,2416,2418],{"class":1124,"line":2415},20,[1122,2417,1471],{"class":1254},[1122,2419,2349],{"class":1135},[1122,2421,2423],{"class":1124,"line":2422},21,[1122,2424,1361],{"class":1135},[1122,2426,2428],{"class":1124,"line":2427},22,[1122,2429,1367],{"class":1135},[1122,2431,2433],{"class":1124,"line":2432},23,[1122,2434,1150],{"emptyLinePlaceholder":1149},[1122,2436,2438],{"class":1124,"line":2437},24,[1122,2439,2440],{"class":1128},"\u002F\u002F 发送事务消息\n",[1122,2442,2444,2446,2449,2451,2454],{"class":1124,"line":2443},25,[1122,2445,1278],{"class":1135},[1122,2447,2448],{"class":1139},"sendMessageInTransaction",[1122,2450,1283],{"class":1135},[1122,2452,2453],{"class":1683},"null",[1122,2455,2262],{"class":1135},[895,2457],{},[903,2459,2461],{"id":2460},"q11-事务消息的回查机制","Q11: 事务消息的回查机制",[892,2463,2464],{},[933,2465,2466],{},"回查触发条件：",[975,2468,2469,2472,2475],{},[978,2470,2471],{},"半消息发送成功，但未收到 Commit\u002FRollback",[978,2473,2474],{},"默认 1 分钟后开始回查",[978,2476,2477],{},"最多回查 15 次",[892,2479,2480],{},[933,2481,2482],{},"本地事务要求：",[975,2484,2485,2492],{},[978,2486,2487,2488,2491],{},"必须是",[933,2489,2490],{},"幂等","的",[978,2493,2494,2495,2498],{},"必须有",[933,2496,2497],{},"状态可查","（如订单状态字段）",[895,2500],{},[898,2502,2504],{"id":2503},"六存储与性能","六、存储与性能",[903,2506,2508],{"id":2507},"q12-为什么-rocketmq-性能高","Q12: 为什么 RocketMQ 性能高",[908,2510,2511,2520],{},[911,2512,2513],{},[914,2514,2515,2518],{},[917,2516,2517],{"align":919},"机制",[917,2519,923],{"align":919},[925,2521,2522,2532,2542,2552],{},[914,2523,2524,2529],{},[930,2525,2526],{"align":919},[933,2527,2528],{},"顺序写",[930,2530,2531],{"align":919},"CommitLog 顺序追加，磁盘顺序写性能高",[914,2533,2534,2539],{},[930,2535,2536],{"align":919},[933,2537,2538],{},"PageCache",[930,2540,2541],{"align":919},"利用操作系统页缓存，减少磁盘 IO",[914,2543,2544,2549],{},[930,2545,2546],{"align":919},[933,2547,2548],{},"零拷贝",[930,2550,2551],{"align":919},"mmap 内存映射，减少数据拷贝",[914,2553,2554,2559],{},[930,2555,2556],{"align":919},[933,2557,2558],{},"异步刷盘",[930,2560,2561],{"align":919},"批量刷盘，提高吞吐量",[895,2563],{},[903,2565,2567],{"id":2566},"q13-同步刷盘-vs-异步刷盘","Q13: 同步刷盘 vs 异步刷盘",[908,2569,2570,2581],{},[911,2571,2572],{},[914,2573,2574,2576,2578],{},[917,2575,1080],{"align":919},[917,2577,923],{"align":919},[917,2579,2580],{"align":919},"特点",[925,2582,2583,2594],{},[914,2584,2585,2588,2591],{},[930,2586,2587],{"align":919},"同步刷盘",[930,2589,2590],{"align":919},"消息写入磁盘后才返回成功",[930,2592,2593],{"align":919},"可靠性高，性能低",[914,2595,2596,2598,2601],{},[930,2597,2558],{"align":919},[930,2599,2600],{"align":919},"消息写入 PageCache 即返回",[930,2602,2603],{"align":919},"性能高，可能丢数据",[1003,2605,2607],{"className":1375,"code":2606,"language":1377,"meta":11,"style":11},"# Broker 配置\nflushDiskType=SYNC_FLUSH   # 同步刷盘\nflushDiskType=ASYNC_FLUSH  # 异步刷盘（默认）\n",[1011,2608,2609,2614,2624],{"__ignoreMap":11},[1122,2610,2611],{"class":1124,"line":1125},[1122,2612,2613],{"class":1128},"# Broker 配置\n",[1122,2615,2616,2618,2621],{"class":1124,"line":1132},[1122,2617,1389],{"class":1254},[1122,2619,2620],{"class":1135},"=SYNC_FLUSH   ",[1122,2622,2623],{"class":1128},"# 同步刷盘\n",[1122,2625,2626,2628,2631],{"class":1124,"line":1146},[1122,2627,1389],{"class":1254},[1122,2629,2630],{"class":1135},"=ASYNC_FLUSH  ",[1122,2632,2633],{"class":1128},"# 异步刷盘（默认）\n",[895,2635],{},[903,2637,2639],{"id":2638},"q14-消息堆积如何处理","Q14: 消息堆积如何处理",[892,2641,2642],{},[933,2643,2644],{},"排查原因：",[2072,2646,2647,2650,2653],{},[978,2648,2649],{},"消费能力不足",[978,2651,2652],{},"消费逻辑有 bug",[978,2654,2655],{},"下游依赖慢",[892,2657,2658],{},[933,2659,2096],{},[908,2661,2662,2670],{},[911,2663,2664],{},[914,2665,2666,2668],{},[917,2667,1548],{"align":919},[917,2669,923],{"align":919},[925,2671,2672,2680,2690,2700,2708],{},[914,2673,2674,2677],{},[930,2675,2676],{"align":919},"扩容消费者",[930,2678,2679],{"align":919},"增加消费实例（不超过 Queue 数）",[914,2681,2682,2685],{},[930,2683,2684],{"align":919},"提高线程数",[930,2686,2687],{"align":919},[1011,2688,2689],{},"consumeThreadMax",[914,2691,2692,2695],{},[930,2693,2694],{"align":919},"批量消费",[930,2696,2697],{"align":919},[1011,2698,2699],{},"consumeMessageBatchMaxSize",[914,2701,2702,2705],{},[930,2703,2704],{"align":919},"临时队列",[930,2706,2707],{"align":919},"紧急情况，消息转存后处理",[914,2709,2710,2713],{},[930,2711,2712],{"align":919},"降级\u002F跳过",[930,2714,2715],{"align":919},"非核心消息直接跳过",[1003,2717,2719],{"className":1116,"code":2718,"language":1118,"meta":11,"style":11},"\u002F\u002F 增加消费线程\nconsumer.setConsumeThreadMax(64);\nconsumer.setConsumeThreadMin(32);\n\n\u002F\u002F 批量消费\nconsumer.setConsumeMessageBatchMaxSize(10);\n",[1011,2720,2721,2726,2740,2754,2758,2763],{"__ignoreMap":11},[1122,2722,2723],{"class":1124,"line":1125},[1122,2724,2725],{"class":1128},"\u002F\u002F 增加消费线程\n",[1122,2727,2728,2730,2733,2735,2738],{"class":1124,"line":1132},[1122,2729,1136],{"class":1135},[1122,2731,2732],{"class":1139},"setConsumeThreadMax",[1122,2734,1664],{"class":1135},[1122,2736,2737],{"class":1683},"64",[1122,2739,2262],{"class":1135},[1122,2741,2742,2744,2747,2749,2752],{"class":1124,"line":1146},[1122,2743,1136],{"class":1135},[1122,2745,2746],{"class":1139},"setConsumeThreadMin",[1122,2748,1664],{"class":1135},[1122,2750,2751],{"class":1683},"32",[1122,2753,2262],{"class":1135},[1122,2755,2756],{"class":1124,"line":1153},[1122,2757,1150],{"emptyLinePlaceholder":1149},[1122,2759,2760],{"class":1124,"line":1159},[1122,2761,2762],{"class":1128},"\u002F\u002F 批量消费\n",[1122,2764,2765,2767,2770,2772,2775],{"class":1124,"line":1295},[1122,2766,1136],{"class":1135},[1122,2768,2769],{"class":1139},"setConsumeMessageBatchMaxSize",[1122,2771,1664],{"class":1135},[1122,2773,2774],{"class":1683},"10",[1122,2776,2262],{"class":1135},[895,2778],{},[898,2780,2782],{"id":2781},"七高可用","七、高可用",[903,2784,2786],{"id":2785},"q15-主从复制模式","Q15: 主从复制模式",[908,2788,2789,2799],{},[911,2790,2791],{},[914,2792,2793,2795,2797],{},[917,2794,1080],{"align":919},[917,2796,923],{"align":919},[917,2798,2580],{"align":919},[925,2800,2801,2812],{},[914,2802,2803,2806,2809],{},[930,2804,2805],{"align":919},"异步复制",[930,2807,2808],{"align":919},"Master 不等 Slave 确认",[930,2810,2811],{"align":919},"性能好，可能丢数据",[914,2813,2814,2817,2820],{},[930,2815,2816],{"align":919},"同步复制",[930,2818,2819],{"align":919},"等待 Slave 确认才返回",[930,2821,2822],{"align":919},"可靠，性能略低",[1003,2824,2826],{"className":1375,"code":2825,"language":1377,"meta":11,"style":11},"# Master 配置\nbrokerRole=ASYNC_MASTER   # 异步复制\nbrokerRole=SYNC_MASTER    # 同步复制\n\n# Slave 配置\nbrokerRole=SLAVE\n",[1011,2827,2828,2833,2843,2853,2857,2862],{"__ignoreMap":11},[1122,2829,2830],{"class":1124,"line":1125},[1122,2831,2832],{"class":1128},"# Master 配置\n",[1122,2834,2835,2837,2840],{"class":1124,"line":1132},[1122,2836,1406],{"class":1254},[1122,2838,2839],{"class":1135},"=ASYNC_MASTER   ",[1122,2841,2842],{"class":1128},"# 异步复制\n",[1122,2844,2845,2847,2850],{"class":1124,"line":1146},[1122,2846,1406],{"class":1254},[1122,2848,2849],{"class":1135},"=SYNC_MASTER    ",[1122,2851,2852],{"class":1128},"# 同步复制\n",[1122,2854,2855],{"class":1124,"line":1153},[1122,2856,1150],{"emptyLinePlaceholder":1149},[1122,2858,2859],{"class":1124,"line":1159},[1122,2860,2861],{"class":1128},"# Slave 配置\n",[1122,2863,2864,2866],{"class":1124,"line":1295},[1122,2865,1406],{"class":1254},[1122,2867,2868],{"class":1135},"=SLAVE\n",[895,2870],{},[903,2872,2874],{"id":2873},"q16-dledger-模式自动选主","Q16: Dledger 模式（自动选主）",[892,2876,2877],{},[933,2878,2879],{},"特点：",[975,2881,2882,2885,2888],{},[978,2883,2884],{},"基于 Raft 协议",[978,2886,2887],{},"自动选举 Master",[978,2889,2890],{},"无需手动切换",[892,2892,2893],{},[933,2894,2895],{},"对比：",[908,2897,2898,2911],{},[911,2899,2900],{},[914,2901,2902,2905,2908],{},[917,2903,2904],{"align":919},"特性",[917,2906,2907],{"align":919},"普通主从",[917,2909,2910],{"align":919},"Dledger",[925,2912,2913,2924,2935],{},[914,2914,2915,2918,2921],{},[930,2916,2917],{"align":919},"选主",[930,2919,2920],{"align":919},"手动",[930,2922,2923],{"align":919},"自动（Raft）",[914,2925,2926,2929,2932],{},[930,2927,2928],{"align":919},"数据一致性",[930,2930,2931],{"align":919},"弱",[930,2933,2934],{"align":919},"强",[914,2936,2937,2940,2943],{},[930,2938,2939],{"align":919},"运维复杂度",[930,2941,2942],{"align":919},"低",[930,2944,2945],{"align":919},"中",[895,2947],{},[898,2949,2951],{"id":2950},"八更多八股文","八、更多八股文",[903,2953,2955],{"id":2954},"q17-消费者负载均衡策略","Q17: 消费者负载均衡策略",[908,2957,2958,2967],{},[911,2959,2960],{},[914,2961,2962,2965],{},[917,2963,2964],{"align":919},"策略",[917,2966,923],{"align":919},[925,2968,2969,2979,2989,2999],{},[914,2970,2971,2976],{},[930,2972,2973],{"align":919},[933,2974,2975],{},"AllocateMessageQueueAveragely",[930,2977,2978],{"align":919},"平均分配（默认）",[914,2980,2981,2986],{},[930,2982,2983],{"align":919},[933,2984,2985],{},"AllocateMessageQueueAveragelyByCircle",[930,2987,2988],{"align":919},"轮询分配",[914,2990,2991,2996],{},[930,2992,2993],{"align":919},[933,2994,2995],{},"AllocateMessageQueueConsistentHash",[930,2997,2998],{"align":919},"一致性哈希",[914,3000,3001,3006],{},[930,3002,3003],{"align":919},[933,3004,3005],{},"AllocateMessageQueueByConfig",[930,3007,3008],{"align":919},"配置指定",[892,3010,3011,3014],{},[933,3012,3013],{},"注意："," 消费者数量不应超过 Queue 数量，否则有消费者空闲。",[895,3016],{},[903,3018,3020],{"id":3019},"q18-消息过滤方式","Q18: 消息过滤方式",[908,3022,3023,3035],{},[911,3024,3025],{},[914,3026,3027,3030,3032],{},[917,3028,3029],{"align":919},"方式",[917,3031,923],{"align":919},[917,3033,3034],{"align":919},"效率",[925,3036,3037,3050,3063],{},[914,3038,3039,3044,3047],{},[930,3040,3041],{"align":919},[933,3042,3043],{},"Tag 过滤",[930,3045,3046],{"align":919},"简单标签过滤",[930,3048,3049],{"align":919},"高（Broker 端）",[914,3051,3052,3057,3060],{},[930,3053,3054],{"align":919},[933,3055,3056],{},"SQL92 过滤",[930,3058,3059],{"align":919},"复杂条件过滤",[930,3061,3062],{"align":919},"中（Broker 端）",[914,3064,3065,3070,3073],{},[930,3066,3067],{"align":919},[933,3068,3069],{},"类过滤",[930,3071,3072],{"align":919},"自定义 Java 类",[930,3074,2942],{"align":919},[1003,3076,3078],{"className":1116,"code":3077,"language":1118,"meta":11,"style":11},"\u002F\u002F Tag 过滤\nconsumer.subscribe(\"Topic\", \"TagA || TagB\");\n\n\u002F\u002F SQL92 过滤\nconsumer.subscribe(\"Topic\", MessageSelector.bySql(\"age > 18\"));\n",[1011,3079,3080,3085,3104,3108,3113],{"__ignoreMap":11},[1122,3081,3082],{"class":1124,"line":1125},[1122,3083,3084],{"class":1128},"\u002F\u002F Tag 过滤\n",[1122,3086,3087,3089,3092,3094,3097,3099,3102],{"class":1124,"line":1132},[1122,3088,1136],{"class":1135},[1122,3090,3091],{"class":1139},"subscribe",[1122,3093,1664],{"class":1135},[1122,3095,3096],{"class":1667},"\"Topic\"",[1122,3098,1680],{"class":1135},[1122,3100,3101],{"class":1667},"\"TagA || TagB\"",[1122,3103,2262],{"class":1135},[1122,3105,3106],{"class":1124,"line":1146},[1122,3107,1150],{"emptyLinePlaceholder":1149},[1122,3109,3110],{"class":1124,"line":1153},[1122,3111,3112],{"class":1128},"\u002F\u002F SQL92 过滤\n",[1122,3114,3115,3117,3119,3121,3123,3126,3129,3131,3134],{"class":1124,"line":1159},[1122,3116,1136],{"class":1135},[1122,3118,3091],{"class":1139},[1122,3120,1664],{"class":1135},[1122,3122,3096],{"class":1667},[1122,3124,3125],{"class":1135},", MessageSelector.",[1122,3127,3128],{"class":1139},"bySql",[1122,3130,1664],{"class":1135},[1122,3132,3133],{"class":1667},"\"age > 18\"",[1122,3135,3136],{"class":1135},"));\n",[895,3138],{},[903,3140,3142],{"id":3141},"q19-消息轨迹","Q19: 消息轨迹",[892,3144,3145,3148],{},[933,3146,3147],{},"作用："," 追踪消息全生命周期",[892,3150,3151],{},[933,3152,3153],{},"包含信息：",[975,3155,3156,3159,3162],{},[978,3157,3158],{},"生产者发送时间、Broker 地址",[978,3160,3161],{},"消费者接收时间、消费结果",[978,3163,3164],{},"重试次数",[1003,3166,3168],{"className":1116,"code":3167,"language":1118,"meta":11,"style":11},"\u002F\u002F 开启消息轨迹\nDefaultMQProducer producer = new DefaultMQProducer(\"group\", true);\n",[1011,3169,3170,3175],{"__ignoreMap":11},[1122,3171,3172],{"class":1124,"line":1125},[1122,3173,3174],{"class":1128},"\u002F\u002F 开启消息轨迹\n",[1122,3176,3177,3180,3182,3184,3187,3189,3191,3193,3196],{"class":1124,"line":1132},[1122,3178,3179],{"class":1135},"DefaultMQProducer producer ",[1122,3181,1255],{"class":1254},[1122,3183,2251],{"class":1254},[1122,3185,3186],{"class":1139}," DefaultMQProducer",[1122,3188,1664],{"class":1135},[1122,3190,2259],{"class":1667},[1122,3192,1680],{"class":1135},[1122,3194,3195],{"class":1683},"true",[1122,3197,2262],{"class":1135},[895,3199],{},[903,3201,3203],{"id":3202},"q20-rocketmq-vs-kafka","Q20: RocketMQ vs Kafka",[908,3205,3206,3218],{},[911,3207,3208],{},[914,3209,3210,3212,3215],{},[917,3211,2904],{"align":919},[917,3213,3214],{"align":919},"RocketMQ",[917,3216,3217],{"align":919},"Kafka",[925,3219,3220,3231,3242,3253,3264,3275,3286],{},[914,3221,3222,3225,3228],{},[930,3223,3224],{"align":919},"开发语言",[930,3226,3227],{"align":919},"Java",[930,3229,3230],{"align":919},"Scala",[914,3232,3233,3236,3239],{},[930,3234,3235],{"align":919},"事务消息",[930,3237,3238],{"align":919},"✅ 原生支持",[930,3240,3241],{"align":919},"0.11+ 支持",[914,3243,3244,3247,3250],{},[930,3245,3246],{"align":919},"延迟消息",[930,3248,3249],{"align":919},"✅ 固定级别 \u002F 5.0 任意",[930,3251,3252],{"align":919},"❌ 需自行实现",[914,3254,3255,3258,3261],{},[930,3256,3257],{"align":919},"消息回溯",[930,3259,3260],{"align":919},"✅ 按时间",[930,3262,3263],{"align":919},"✅ 按 offset",[914,3265,3266,3269,3272],{},[930,3267,3268],{"align":919},"消息过滤",[930,3270,3271],{"align":919},"✅ Tag\u002FSQL",[930,3273,3274],{"align":919},"❌",[914,3276,3277,3280,3283],{},[930,3278,3279],{"align":919},"吞吐量",[930,3281,3282],{"align":919},"十万级",[930,3284,3285],{"align":919},"百万级",[914,3287,3288,3290,3293],{},[930,3289,1554],{"align":919},[930,3291,3292],{"align":919},"业务消息",[930,3294,3295],{"align":919},"日志、大数据",[895,3297],{},[903,3299,3301],{"id":3300},"q21-nameserver-vs-zookeeper","Q21: NameServer vs ZooKeeper",[908,3303,3304,3315],{},[911,3305,3306],{},[914,3307,3308,3310,3312],{},[917,3309,2904],{"align":919},[917,3311,935],{"align":919},[917,3313,3314],{"align":919},"ZooKeeper",[925,3316,3317,3328,3339,3350],{},[914,3318,3319,3322,3325],{},[930,3320,3321],{"align":919},"架构",[930,3323,3324],{"align":919},"无状态，互不通信",[930,3326,3327],{"align":919},"有状态，需要选主",[914,3329,3330,3333,3336],{},[930,3331,3332],{"align":919},"一致性",[930,3334,3335],{"align":919},"最终一致",[930,3337,3338],{"align":919},"强一致（ZAB）",[914,3340,3341,3344,3347],{},[930,3342,3343],{"align":919},"复杂度",[930,3345,3346],{"align":919},"简单",[930,3348,3349],{"align":919},"复杂",[914,3351,3352,3355,3358],{},[930,3353,3354],{"align":919},"功能",[930,3356,3357],{"align":919},"仅路由发现",[930,3359,3360],{"align":919},"配置、选主、分布式锁",[892,3362,3363,3366],{},[933,3364,3365],{},"为什么选 NameServer："," 简单、轻量、足够用。",[895,3368],{},[903,3370,3372],{"id":3371},"q22-消费位点管理","Q22: 消费位点管理",[892,3374,3375],{},[933,3376,3377],{},"存储位置：",[975,3379,3380,3385],{},[978,3381,3382,3384],{},[933,3383,1094],{},"：Broker 端存储",[978,3386,3387,3389],{},[933,3388,1107],{},"：消费者本地存储",[892,3391,3392],{},[933,3393,3394],{},"位点提交方式：",[975,3396,3397,3403],{},[978,3398,3399,3402],{},[933,3400,3401],{},"自动提交","：定时自动提交",[978,3404,3405,3408],{},[933,3406,3407],{},"手动提交","：消费成功后手动提交",[1003,3410,3412],{"className":1116,"code":3411,"language":1118,"meta":11,"style":11},"\u002F\u002F 手动提交（推荐）\nconsumer.setAutoCommit(false);\nconsumer.commitSync();\n",[1011,3413,3414,3419,3433],{"__ignoreMap":11},[1122,3415,3416],{"class":1124,"line":1125},[1122,3417,3418],{"class":1128},"\u002F\u002F 手动提交（推荐）\n",[1122,3420,3421,3423,3426,3428,3431],{"class":1124,"line":1132},[1122,3422,1136],{"class":1135},[1122,3424,3425],{"class":1139},"setAutoCommit",[1122,3427,1664],{"class":1135},[1122,3429,3430],{"class":1683},"false",[1122,3432,2262],{"class":1135},[1122,3434,3435,3437,3440],{"class":1124,"line":1146},[1122,3436,1136],{"class":1135},[1122,3438,3439],{"class":1139},"commitSync",[1122,3441,1640],{"class":1135},[895,3443],{},[898,3445,3447],{"id":3446},"九高频考点清单","九、高频考点清单",[903,3449,3450],{"id":3450},"必考",[975,3452,3455,3464,3470,3476,3482],{"className":3453},[3454],"contains-task-list",[978,3456,3459,3463],{"className":3457},[3458],"task-list-item",[3460,3461],"input",{"disabled":1149,"type":3462},"checkbox"," 核心组件和架构",[978,3465,3467,3469],{"className":3466},[3458],[3460,3468],{"disabled":1149,"type":3462}," 消息不丢失的保障（三个环节）",[978,3471,3473,3475],{"className":3472},[3458],[3460,3474],{"disabled":1149,"type":3462}," 幂等性实现方案",[978,3477,3479,3481],{"className":3478},[3458],[3460,3480],{"disabled":1149,"type":3462}," 顺序消息实现",[978,3483,3485,3487],{"className":3484},[3458],[3460,3486],{"disabled":1149,"type":3462}," 事务消息流程和回查",[903,3489,3490],{"id":3490},"常考",[975,3492,3494,3500,3506,3512,3518,3524],{"className":3493},[3454],[978,3495,3497,3499],{"className":3496},[3458],[3460,3498],{"disabled":1149,"type":3462}," CommitLog\u002FConsumeQueue\u002FIndexFile 作用",[978,3501,3503,3505],{"className":3502},[3458],[3460,3504],{"disabled":1149,"type":3462}," 延迟消息实现",[978,3507,3509,3511],{"className":3508},[3458],[3460,3510],{"disabled":1149,"type":3462}," 消费重试和死信队列",[978,3513,3515,3517],{"className":3514},[3458],[3460,3516],{"disabled":1149,"type":3462}," 同步\u002F异步刷盘区别",[978,3519,3521,3523],{"className":3520},[3458],[3460,3522],{"disabled":1149,"type":3462}," 消息堆积处理",[978,3525,3527,3529],{"className":3526},[3458],[3460,3528],{"disabled":1149,"type":3462}," 消费者负载均衡策略",[903,3531,3532],{"id":3532},"进阶",[975,3534,3536,3542,3548,3554,3560],{"className":3535},[3454],[978,3537,3539,3541],{"className":3538},[3458],[3460,3540],{"disabled":1149,"type":3462}," 同步\u002F异步复制区别",[978,3543,3545,3547],{"className":3544},[3458],[3460,3546],{"disabled":1149,"type":3462}," Dledger 模式",[978,3549,3551,3553],{"className":3550},[3458],[3460,3552],{"disabled":1149,"type":3462}," 顺序写 + PageCache + 零拷贝",[978,3555,3557,3559],{"className":3556},[3458],[3460,3558],{"disabled":1149,"type":3462}," RocketMQ vs Kafka",[978,3561,3563,3565],{"className":3562},[3458],[3460,3564],{"disabled":1149,"type":3462}," NameServer vs ZooKeeper",[3567,3568,3569],"style",{},"html pre.shiki code .sCsY4, html code.shiki .sCsY4{--shiki-light:#6A737D;--shiki-default:#6A737D;--shiki-dark:#6A737D}html pre.shiki code .sxrX7, html code.shiki .sxrX7{--shiki-light:#24292E;--shiki-default:#24292E;--shiki-dark:#E1E4E8}html pre.shiki code .snPdu, html code.shiki .snPdu{--shiki-light:#6F42C1;--shiki-default:#6F42C1;--shiki-dark:#B392F0}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 .sP4rz, html code.shiki .sP4rz{--shiki-light:#E36209;--shiki-default:#E36209;--shiki-dark:#FFAB70}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}",{"title":11,"searchDepth":1132,"depth":1132,"links":3571},[3572,3577,3582,3586,3589,3593,3598,3602,3610],{"id":900,"depth":1132,"text":901,"children":3573},[3574,3575,3576],{"id":905,"depth":1146,"text":906},{"id":1000,"depth":1146,"text":1001},{"id":1070,"depth":1146,"text":1071},{"id":1171,"depth":1132,"text":1172,"children":3578},[3579,3580,3581],{"id":1175,"depth":1146,"text":1176},{"id":1414,"depth":1146,"text":1415},{"id":1533,"depth":1146,"text":1534},{"id":1796,"depth":1132,"text":1797,"children":3583},[3584,3585],{"id":1800,"depth":1146,"text":1801},{"id":2064,"depth":1146,"text":2065},{"id":2109,"depth":1132,"text":2110,"children":3587},[3588],{"id":2113,"depth":1146,"text":2114},{"id":2220,"depth":1132,"text":2221,"children":3590},[3591,3592],{"id":2224,"depth":1146,"text":2225},{"id":2460,"depth":1146,"text":2461},{"id":2503,"depth":1132,"text":2504,"children":3594},[3595,3596,3597],{"id":2507,"depth":1146,"text":2508},{"id":2566,"depth":1146,"text":2567},{"id":2638,"depth":1146,"text":2639},{"id":2781,"depth":1132,"text":2782,"children":3599},[3600,3601],{"id":2785,"depth":1146,"text":2786},{"id":2873,"depth":1146,"text":2874},{"id":2950,"depth":1132,"text":2951,"children":3603},[3604,3605,3606,3607,3608,3609],{"id":2954,"depth":1146,"text":2955},{"id":3019,"depth":1146,"text":3020},{"id":3141,"depth":1146,"text":3142},{"id":3202,"depth":1146,"text":3203},{"id":3300,"depth":1146,"text":3301},{"id":3371,"depth":1146,"text":3372},{"id":3446,"depth":1132,"text":3447,"children":3611},[3612,3613,3614],{"id":3450,"depth":1146,"text":3450},{"id":3490,"depth":1146,"text":3490},{"id":3532,"depth":1146,"text":3532},"md",{},{"title":354,"description":355},"interview\u002Frocketmq","zuMgKBkj_LXwuqE5xe8aaJL5y7CWcWpzIhqAwiTfzAs",1775496433516]