[{"data":1,"prerenderedAt":3338},["ShallowReactive",2],{"search-docs":3,"doc-\u002Fother\u002Fspring-series\u002Fspringmvc\u002Fspringmvc-interceptor":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":565,"body":888,"description":11,"extension":3332,"meta":3333,"navigation":3334,"path":564,"seo":3335,"stem":3336,"__hash__":3337},"docs\u002Fother\u002Fspring-series\u002Fspringmvc\u002Fspringmvc-interceptor.md",{"type":889,"value":890,"toc":3312},"minimark",[891,895,899,904,972,975,985,988,992,1499,1502,1506,1633,1637,1859,1862,1865,2078,2082,2504,2507,3055,3058,3278,3281,3284,3290,3303,3308],[892,893,894],"h2",{"id":894},"概述",[896,897,898],"p",{},"拦截器（Interceptor）是 Spring MVC 提供的一种机制，用于在请求处理前后执行特定逻辑，类似于 Servlet 的 Filter，但更加灵活。",[900,901,903],"h3",{"id":902},"拦截器-vs-过滤器","拦截器 vs 过滤器",[905,906,907,924],"table",{},[908,909,910],"thead",{},[911,912,913,918,921],"tr",{},[914,915,917],"th",{"align":916},"left","特性",[914,919,920],{"align":916},"拦截器（Interceptor）",[914,922,923],{"align":916},"过滤器（Filter）",[925,926,927,939,950,961],"tbody",{},[911,928,929,933,936],{},[930,931,932],"td",{"align":916},"规范",[930,934,935],{"align":916},"Spring MVC",[930,937,938],{"align":916},"Servlet",[911,940,941,944,947],{},[930,942,943],{"align":916},"使用范围",[930,945,946],{"align":916},"只能拦截 Controller",[930,948,949],{"align":916},"可以拦截所有请求",[911,951,952,955,958],{},[930,953,954],{"align":916},"获取 Bean",[930,956,957],{"align":916},"可以获取 Spring Bean",[930,959,960],{"align":916},"不能直接获取",[911,962,963,966,969],{},[930,964,965],{"align":916},"执行顺序",[930,967,968],{"align":916},"在 DispatcherServlet 之后",[930,970,971],{"align":916},"在 DispatcherServlet 之前",[900,973,974],{"id":974},"执行流程",[976,977,982],"pre",{"className":978,"code":980,"language":981},[979],"language-text","请求 → Filter → DispatcherServlet → Interceptor.preHandle\n                                          ↓\n                                     Controller\n                                          ↓\n                                   Interceptor.postHandle\n                                          ↓\n                                     视图渲染\n                                          ↓\n                                Interceptor.afterCompletion\n                                          ↓\n响应 ← Filter ← DispatcherServlet ←────────┘\n","text",[983,984,980],"code",{"__ignoreMap":11},[892,986,987],{"id":987},"创建拦截器",[900,989,991],{"id":990},"实现-handlerinterceptor-接口","实现 HandlerInterceptor 接口",[976,993,997],{"className":994,"code":995,"language":996,"meta":11,"style":11},"language-java shiki shiki-themes github-light github-light github-dark","@Component\npublic class MyInterceptor implements HandlerInterceptor {\n    \n    \u002F**\n     * 请求处理之前调用\n     * @return true 放行，false 拦截\n     *\u002F\n    @Override\n    public boolean preHandle(HttpServletRequest request, \n                            HttpServletResponse response, \n                            Object handler) throws Exception {\n        System.out.println(\"preHandle: 请求处理前\");\n        \n        \u002F\u002F 获取请求信息\n        String uri = request.getRequestURI();\n        String method = request.getMethod();\n        \n        \u002F\u002F 可以进行权限验证、登录检查等\n        return true;  \u002F\u002F 放行\n    }\n    \n    \u002F**\n     * 请求处理之后，视图渲染之前调用\n     *\u002F\n    @Override\n    public void postHandle(HttpServletRequest request, \n                          HttpServletResponse response, \n                          Object handler, \n                          ModelAndView modelAndView) throws Exception {\n        System.out.println(\"postHandle: 请求处理后\");\n        \n        \u002F\u002F 可以修改 ModelAndView\n        if (modelAndView != null) {\n            modelAndView.addObject(\"interceptorData\", \"拦截器数据\");\n        }\n    }\n    \n    \u002F**\n     * 整个请求完成后调用（视图渲染后）\n     *\u002F\n    @Override\n    public void afterCompletion(HttpServletRequest request, \n                               HttpServletResponse response, \n                               Object handler, \n                               Exception ex) throws Exception {\n        System.out.println(\"afterCompletion: 请求完成\");\n        \n        \u002F\u002F 可以进行资源清理、日志记录等\n    }\n}\n","java",[983,998,999,1012,1034,1040,1047,1053,1065,1071,1080,1102,1113,1131,1150,1156,1162,1180,1195,1200,1206,1222,1228,1233,1238,1244,1249,1256,1273,1283,1293,1308,1322,1327,1333,1351,1373,1379,1384,1389,1394,1400,1405,1412,1428,1438,1448,1463,1477,1482,1488,1493],{"__ignoreMap":11},[1000,1001,1004,1008],"span",{"class":1002,"line":1003},"line",1,[1000,1005,1007],{"class":1006},"sxrX7","@",[1000,1009,1011],{"class":1010},"s8jYJ","Component\n",[1000,1013,1015,1018,1021,1025,1028,1031],{"class":1002,"line":1014},2,[1000,1016,1017],{"class":1010},"public",[1000,1019,1020],{"class":1010}," class",[1000,1022,1024],{"class":1023},"snPdu"," MyInterceptor",[1000,1026,1027],{"class":1010}," implements",[1000,1029,1030],{"class":1023}," HandlerInterceptor",[1000,1032,1033],{"class":1006}," {\n",[1000,1035,1037],{"class":1002,"line":1036},3,[1000,1038,1039],{"class":1006},"    \n",[1000,1041,1043],{"class":1002,"line":1042},4,[1000,1044,1046],{"class":1045},"sCsY4","    \u002F**\n",[1000,1048,1050],{"class":1002,"line":1049},5,[1000,1051,1052],{"class":1045},"     * 请求处理之前调用\n",[1000,1054,1056,1059,1062],{"class":1002,"line":1055},6,[1000,1057,1058],{"class":1045},"     * ",[1000,1060,1061],{"class":1010},"@return",[1000,1063,1064],{"class":1045}," true 放行，false 拦截\n",[1000,1066,1068],{"class":1002,"line":1067},7,[1000,1069,1070],{"class":1045},"     *\u002F\n",[1000,1072,1074,1077],{"class":1002,"line":1073},8,[1000,1075,1076],{"class":1006},"    @",[1000,1078,1079],{"class":1010},"Override\n",[1000,1081,1083,1086,1089,1092,1095,1099],{"class":1002,"line":1082},9,[1000,1084,1085],{"class":1010},"    public",[1000,1087,1088],{"class":1010}," boolean",[1000,1090,1091],{"class":1023}," preHandle",[1000,1093,1094],{"class":1006},"(HttpServletRequest ",[1000,1096,1098],{"class":1097},"sP4rz","request",[1000,1100,1101],{"class":1006},", \n",[1000,1103,1105,1108,1111],{"class":1002,"line":1104},10,[1000,1106,1107],{"class":1006},"                            HttpServletResponse ",[1000,1109,1110],{"class":1097},"response",[1000,1112,1101],{"class":1006},[1000,1114,1116,1119,1122,1125,1128],{"class":1002,"line":1115},11,[1000,1117,1118],{"class":1006},"                            Object ",[1000,1120,1121],{"class":1097},"handler",[1000,1123,1124],{"class":1006},") ",[1000,1126,1127],{"class":1010},"throws",[1000,1129,1130],{"class":1006}," Exception {\n",[1000,1132,1134,1137,1140,1143,1147],{"class":1002,"line":1133},12,[1000,1135,1136],{"class":1006},"        System.out.",[1000,1138,1139],{"class":1023},"println",[1000,1141,1142],{"class":1006},"(",[1000,1144,1146],{"class":1145},"sIIMD","\"preHandle: 请求处理前\"",[1000,1148,1149],{"class":1006},");\n",[1000,1151,1153],{"class":1002,"line":1152},13,[1000,1154,1155],{"class":1006},"        \n",[1000,1157,1159],{"class":1002,"line":1158},14,[1000,1160,1161],{"class":1045},"        \u002F\u002F 获取请求信息\n",[1000,1163,1165,1168,1171,1174,1177],{"class":1002,"line":1164},15,[1000,1166,1167],{"class":1006},"        String uri ",[1000,1169,1170],{"class":1010},"=",[1000,1172,1173],{"class":1006}," request.",[1000,1175,1176],{"class":1023},"getRequestURI",[1000,1178,1179],{"class":1006},"();\n",[1000,1181,1183,1186,1188,1190,1193],{"class":1002,"line":1182},16,[1000,1184,1185],{"class":1006},"        String method ",[1000,1187,1170],{"class":1010},[1000,1189,1173],{"class":1006},[1000,1191,1192],{"class":1023},"getMethod",[1000,1194,1179],{"class":1006},[1000,1196,1198],{"class":1002,"line":1197},17,[1000,1199,1155],{"class":1006},[1000,1201,1203],{"class":1002,"line":1202},18,[1000,1204,1205],{"class":1045},"        \u002F\u002F 可以进行权限验证、登录检查等\n",[1000,1207,1209,1212,1216,1219],{"class":1002,"line":1208},19,[1000,1210,1211],{"class":1010},"        return",[1000,1213,1215],{"class":1214},"sBjJW"," true",[1000,1217,1218],{"class":1006},";  ",[1000,1220,1221],{"class":1045},"\u002F\u002F 放行\n",[1000,1223,1225],{"class":1002,"line":1224},20,[1000,1226,1227],{"class":1006},"    }\n",[1000,1229,1231],{"class":1002,"line":1230},21,[1000,1232,1039],{"class":1006},[1000,1234,1236],{"class":1002,"line":1235},22,[1000,1237,1046],{"class":1045},[1000,1239,1241],{"class":1002,"line":1240},23,[1000,1242,1243],{"class":1045},"     * 请求处理之后，视图渲染之前调用\n",[1000,1245,1247],{"class":1002,"line":1246},24,[1000,1248,1070],{"class":1045},[1000,1250,1252,1254],{"class":1002,"line":1251},25,[1000,1253,1076],{"class":1006},[1000,1255,1079],{"class":1010},[1000,1257,1259,1261,1264,1267,1269,1271],{"class":1002,"line":1258},26,[1000,1260,1085],{"class":1010},[1000,1262,1263],{"class":1010}," void",[1000,1265,1266],{"class":1023}," postHandle",[1000,1268,1094],{"class":1006},[1000,1270,1098],{"class":1097},[1000,1272,1101],{"class":1006},[1000,1274,1276,1279,1281],{"class":1002,"line":1275},27,[1000,1277,1278],{"class":1006},"                          HttpServletResponse ",[1000,1280,1110],{"class":1097},[1000,1282,1101],{"class":1006},[1000,1284,1286,1289,1291],{"class":1002,"line":1285},28,[1000,1287,1288],{"class":1006},"                          Object ",[1000,1290,1121],{"class":1097},[1000,1292,1101],{"class":1006},[1000,1294,1296,1299,1302,1304,1306],{"class":1002,"line":1295},29,[1000,1297,1298],{"class":1006},"                          ModelAndView ",[1000,1300,1301],{"class":1097},"modelAndView",[1000,1303,1124],{"class":1006},[1000,1305,1127],{"class":1010},[1000,1307,1130],{"class":1006},[1000,1309,1311,1313,1315,1317,1320],{"class":1002,"line":1310},30,[1000,1312,1136],{"class":1006},[1000,1314,1139],{"class":1023},[1000,1316,1142],{"class":1006},[1000,1318,1319],{"class":1145},"\"postHandle: 请求处理后\"",[1000,1321,1149],{"class":1006},[1000,1323,1325],{"class":1002,"line":1324},31,[1000,1326,1155],{"class":1006},[1000,1328,1330],{"class":1002,"line":1329},32,[1000,1331,1332],{"class":1045},"        \u002F\u002F 可以修改 ModelAndView\n",[1000,1334,1336,1339,1342,1345,1348],{"class":1002,"line":1335},33,[1000,1337,1338],{"class":1010},"        if",[1000,1340,1341],{"class":1006}," (modelAndView ",[1000,1343,1344],{"class":1010},"!=",[1000,1346,1347],{"class":1214}," null",[1000,1349,1350],{"class":1006},") {\n",[1000,1352,1354,1357,1360,1362,1365,1368,1371],{"class":1002,"line":1353},34,[1000,1355,1356],{"class":1006},"            modelAndView.",[1000,1358,1359],{"class":1023},"addObject",[1000,1361,1142],{"class":1006},[1000,1363,1364],{"class":1145},"\"interceptorData\"",[1000,1366,1367],{"class":1006},", ",[1000,1369,1370],{"class":1145},"\"拦截器数据\"",[1000,1372,1149],{"class":1006},[1000,1374,1376],{"class":1002,"line":1375},35,[1000,1377,1378],{"class":1006},"        }\n",[1000,1380,1382],{"class":1002,"line":1381},36,[1000,1383,1227],{"class":1006},[1000,1385,1387],{"class":1002,"line":1386},37,[1000,1388,1039],{"class":1006},[1000,1390,1392],{"class":1002,"line":1391},38,[1000,1393,1046],{"class":1045},[1000,1395,1397],{"class":1002,"line":1396},39,[1000,1398,1399],{"class":1045},"     * 整个请求完成后调用（视图渲染后）\n",[1000,1401,1403],{"class":1002,"line":1402},40,[1000,1404,1070],{"class":1045},[1000,1406,1408,1410],{"class":1002,"line":1407},41,[1000,1409,1076],{"class":1006},[1000,1411,1079],{"class":1010},[1000,1413,1415,1417,1419,1422,1424,1426],{"class":1002,"line":1414},42,[1000,1416,1085],{"class":1010},[1000,1418,1263],{"class":1010},[1000,1420,1421],{"class":1023}," afterCompletion",[1000,1423,1094],{"class":1006},[1000,1425,1098],{"class":1097},[1000,1427,1101],{"class":1006},[1000,1429,1431,1434,1436],{"class":1002,"line":1430},43,[1000,1432,1433],{"class":1006},"                               HttpServletResponse ",[1000,1435,1110],{"class":1097},[1000,1437,1101],{"class":1006},[1000,1439,1441,1444,1446],{"class":1002,"line":1440},44,[1000,1442,1443],{"class":1006},"                               Object ",[1000,1445,1121],{"class":1097},[1000,1447,1101],{"class":1006},[1000,1449,1451,1454,1457,1459,1461],{"class":1002,"line":1450},45,[1000,1452,1453],{"class":1006},"                               Exception ",[1000,1455,1456],{"class":1097},"ex",[1000,1458,1124],{"class":1006},[1000,1460,1127],{"class":1010},[1000,1462,1130],{"class":1006},[1000,1464,1466,1468,1470,1472,1475],{"class":1002,"line":1465},46,[1000,1467,1136],{"class":1006},[1000,1469,1139],{"class":1023},[1000,1471,1142],{"class":1006},[1000,1473,1474],{"class":1145},"\"afterCompletion: 请求完成\"",[1000,1476,1149],{"class":1006},[1000,1478,1480],{"class":1002,"line":1479},47,[1000,1481,1155],{"class":1006},[1000,1483,1485],{"class":1002,"line":1484},48,[1000,1486,1487],{"class":1045},"        \u002F\u002F 可以进行资源清理、日志记录等\n",[1000,1489,1491],{"class":1002,"line":1490},49,[1000,1492,1227],{"class":1006},[1000,1494,1496],{"class":1002,"line":1495},50,[1000,1497,1498],{"class":1006},"}\n",[892,1500,1501],{"id":1501},"注册拦截器",[900,1503,1505],{"id":1504},"xml-配置","XML 配置",[976,1507,1511],{"className":1508,"code":1509,"language":1510,"meta":11,"style":11},"language-xml shiki shiki-themes github-light github-light github-dark","\u003Cmvc:interceptors>\n    \u003C!-- 拦截所有请求 -->\n    \u003Cbean class=\"com.example.interceptor.MyInterceptor\"\u002F>\n    \n    \u003C!-- 指定拦截路径 -->\n    \u003Cmvc:interceptor>\n        \u003Cmvc:mapping path=\"\u002Fadmin\u002F**\"\u002F>\n        \u003Cmvc:exclude-mapping path=\"\u002Fadmin\u002Flogin\"\u002F>\n        \u003Cbean class=\"com.example.interceptor.AdminInterceptor\"\u002F>\n    \u003C\u002Fmvc:interceptor>\n\u003C\u002Fmvc:interceptors>\n","xml",[983,1512,1513,1525,1530,1548,1552,1557,1566,1584,1600,1615,1624],{"__ignoreMap":11},[1000,1514,1515,1518,1522],{"class":1002,"line":1003},[1000,1516,1517],{"class":1006},"\u003C",[1000,1519,1521],{"class":1520},"sovSZ","mvc:interceptors",[1000,1523,1524],{"class":1006},">\n",[1000,1526,1527],{"class":1002,"line":1014},[1000,1528,1529],{"class":1045},"    \u003C!-- 拦截所有请求 -->\n",[1000,1531,1532,1535,1538,1540,1542,1545],{"class":1002,"line":1036},[1000,1533,1534],{"class":1006},"    \u003C",[1000,1536,1537],{"class":1520},"bean",[1000,1539,1020],{"class":1023},[1000,1541,1170],{"class":1006},[1000,1543,1544],{"class":1145},"\"com.example.interceptor.MyInterceptor\"",[1000,1546,1547],{"class":1006},"\u002F>\n",[1000,1549,1550],{"class":1002,"line":1042},[1000,1551,1039],{"class":1006},[1000,1553,1554],{"class":1002,"line":1049},[1000,1555,1556],{"class":1045},"    \u003C!-- 指定拦截路径 -->\n",[1000,1558,1559,1561,1564],{"class":1002,"line":1055},[1000,1560,1534],{"class":1006},[1000,1562,1563],{"class":1520},"mvc:interceptor",[1000,1565,1524],{"class":1006},[1000,1567,1568,1571,1574,1577,1579,1582],{"class":1002,"line":1067},[1000,1569,1570],{"class":1006},"        \u003C",[1000,1572,1573],{"class":1520},"mvc:mapping",[1000,1575,1576],{"class":1023}," path",[1000,1578,1170],{"class":1006},[1000,1580,1581],{"class":1145},"\"\u002Fadmin\u002F**\"",[1000,1583,1547],{"class":1006},[1000,1585,1586,1588,1591,1593,1595,1598],{"class":1002,"line":1073},[1000,1587,1570],{"class":1006},[1000,1589,1590],{"class":1520},"mvc:exclude-mapping",[1000,1592,1576],{"class":1023},[1000,1594,1170],{"class":1006},[1000,1596,1597],{"class":1145},"\"\u002Fadmin\u002Flogin\"",[1000,1599,1547],{"class":1006},[1000,1601,1602,1604,1606,1608,1610,1613],{"class":1002,"line":1082},[1000,1603,1570],{"class":1006},[1000,1605,1537],{"class":1520},[1000,1607,1020],{"class":1023},[1000,1609,1170],{"class":1006},[1000,1611,1612],{"class":1145},"\"com.example.interceptor.AdminInterceptor\"",[1000,1614,1547],{"class":1006},[1000,1616,1617,1620,1622],{"class":1002,"line":1104},[1000,1618,1619],{"class":1006},"    \u003C\u002F",[1000,1621,1563],{"class":1520},[1000,1623,1524],{"class":1006},[1000,1625,1626,1629,1631],{"class":1002,"line":1115},[1000,1627,1628],{"class":1006},"\u003C\u002F",[1000,1630,1521],{"class":1520},[1000,1632,1524],{"class":1006},[900,1634,1636],{"id":1635},"java-配置","Java 配置",[976,1638,1640],{"className":994,"code":1639,"language":996,"meta":11,"style":11},"@Configuration\npublic class WebConfig implements WebMvcConfigurer {\n    \n    @Autowired\n    private LoginInterceptor loginInterceptor;\n    \n    @Autowired\n    private AdminInterceptor adminInterceptor;\n    \n    @Override\n    public void addInterceptors(InterceptorRegistry registry) {\n        \u002F\u002F 登录拦截器\n        registry.addInterceptor(loginInterceptor)\n                .addPathPatterns(\"\u002F**\")  \u002F\u002F 拦截所有\n                .excludePathPatterns(    \u002F\u002F 排除路径\n                    \"\u002Flogin\",\n                    \"\u002Fregister\",\n                    \"\u002Fstatic\u002F**\",\n                    \"\u002Ferror\"\n                );\n        \n        \u002F\u002F 管理员拦截器\n        registry.addInterceptor(adminInterceptor)\n                .addPathPatterns(\"\u002Fadmin\u002F**\")\n                .excludePathPatterns(\"\u002Fadmin\u002Flogin\");\n    }\n}\n",[983,1641,1642,1649,1665,1669,1676,1684,1688,1694,1701,1705,1711,1728,1733,1744,1763,1776,1784,1791,1798,1803,1808,1812,1817,1826,1839,1851,1855],{"__ignoreMap":11},[1000,1643,1644,1646],{"class":1002,"line":1003},[1000,1645,1007],{"class":1006},[1000,1647,1648],{"class":1010},"Configuration\n",[1000,1650,1651,1653,1655,1658,1660,1663],{"class":1002,"line":1014},[1000,1652,1017],{"class":1010},[1000,1654,1020],{"class":1010},[1000,1656,1657],{"class":1023}," WebConfig",[1000,1659,1027],{"class":1010},[1000,1661,1662],{"class":1023}," WebMvcConfigurer",[1000,1664,1033],{"class":1006},[1000,1666,1667],{"class":1002,"line":1036},[1000,1668,1039],{"class":1006},[1000,1670,1671,1673],{"class":1002,"line":1042},[1000,1672,1076],{"class":1006},[1000,1674,1675],{"class":1010},"Autowired\n",[1000,1677,1678,1681],{"class":1002,"line":1049},[1000,1679,1680],{"class":1010},"    private",[1000,1682,1683],{"class":1006}," LoginInterceptor loginInterceptor;\n",[1000,1685,1686],{"class":1002,"line":1055},[1000,1687,1039],{"class":1006},[1000,1689,1690,1692],{"class":1002,"line":1067},[1000,1691,1076],{"class":1006},[1000,1693,1675],{"class":1010},[1000,1695,1696,1698],{"class":1002,"line":1073},[1000,1697,1680],{"class":1010},[1000,1699,1700],{"class":1006}," AdminInterceptor adminInterceptor;\n",[1000,1702,1703],{"class":1002,"line":1082},[1000,1704,1039],{"class":1006},[1000,1706,1707,1709],{"class":1002,"line":1104},[1000,1708,1076],{"class":1006},[1000,1710,1079],{"class":1010},[1000,1712,1713,1715,1717,1720,1723,1726],{"class":1002,"line":1115},[1000,1714,1085],{"class":1010},[1000,1716,1263],{"class":1010},[1000,1718,1719],{"class":1023}," addInterceptors",[1000,1721,1722],{"class":1006},"(InterceptorRegistry ",[1000,1724,1725],{"class":1097},"registry",[1000,1727,1350],{"class":1006},[1000,1729,1730],{"class":1002,"line":1133},[1000,1731,1732],{"class":1045},"        \u002F\u002F 登录拦截器\n",[1000,1734,1735,1738,1741],{"class":1002,"line":1152},[1000,1736,1737],{"class":1006},"        registry.",[1000,1739,1740],{"class":1023},"addInterceptor",[1000,1742,1743],{"class":1006},"(loginInterceptor)\n",[1000,1745,1746,1749,1752,1754,1757,1760],{"class":1002,"line":1158},[1000,1747,1748],{"class":1006},"                .",[1000,1750,1751],{"class":1023},"addPathPatterns",[1000,1753,1142],{"class":1006},[1000,1755,1756],{"class":1145},"\"\u002F**\"",[1000,1758,1759],{"class":1006},")  ",[1000,1761,1762],{"class":1045},"\u002F\u002F 拦截所有\n",[1000,1764,1765,1767,1770,1773],{"class":1002,"line":1164},[1000,1766,1748],{"class":1006},[1000,1768,1769],{"class":1023},"excludePathPatterns",[1000,1771,1772],{"class":1006},"(    ",[1000,1774,1775],{"class":1045},"\u002F\u002F 排除路径\n",[1000,1777,1778,1781],{"class":1002,"line":1182},[1000,1779,1780],{"class":1145},"                    \"\u002Flogin\"",[1000,1782,1783],{"class":1006},",\n",[1000,1785,1786,1789],{"class":1002,"line":1197},[1000,1787,1788],{"class":1145},"                    \"\u002Fregister\"",[1000,1790,1783],{"class":1006},[1000,1792,1793,1796],{"class":1002,"line":1202},[1000,1794,1795],{"class":1145},"                    \"\u002Fstatic\u002F**\"",[1000,1797,1783],{"class":1006},[1000,1799,1800],{"class":1002,"line":1208},[1000,1801,1802],{"class":1145},"                    \"\u002Ferror\"\n",[1000,1804,1805],{"class":1002,"line":1224},[1000,1806,1807],{"class":1006},"                );\n",[1000,1809,1810],{"class":1002,"line":1230},[1000,1811,1155],{"class":1006},[1000,1813,1814],{"class":1002,"line":1235},[1000,1815,1816],{"class":1045},"        \u002F\u002F 管理员拦截器\n",[1000,1818,1819,1821,1823],{"class":1002,"line":1240},[1000,1820,1737],{"class":1006},[1000,1822,1740],{"class":1023},[1000,1824,1825],{"class":1006},"(adminInterceptor)\n",[1000,1827,1828,1830,1832,1834,1836],{"class":1002,"line":1246},[1000,1829,1748],{"class":1006},[1000,1831,1751],{"class":1023},[1000,1833,1142],{"class":1006},[1000,1835,1581],{"class":1145},[1000,1837,1838],{"class":1006},")\n",[1000,1840,1841,1843,1845,1847,1849],{"class":1002,"line":1251},[1000,1842,1748],{"class":1006},[1000,1844,1769],{"class":1023},[1000,1846,1142],{"class":1006},[1000,1848,1597],{"class":1145},[1000,1850,1149],{"class":1006},[1000,1852,1853],{"class":1002,"line":1258},[1000,1854,1227],{"class":1006},[1000,1856,1857],{"class":1002,"line":1275},[1000,1858,1498],{"class":1006},[892,1860,1861],{"id":1861},"实战示例",[900,1863,1864],{"id":1864},"登录拦截器",[976,1866,1868],{"className":994,"code":1867,"language":996,"meta":11,"style":11},"@Component\npublic class LoginInterceptor implements HandlerInterceptor {\n    \n    @Override\n    public boolean preHandle(HttpServletRequest request, \n                            HttpServletResponse response, \n                            Object handler) throws Exception {\n        \u002F\u002F 获取 Session 中的用户信息\n        HttpSession session = request.getSession();\n        Object user = session.getAttribute(\"loginUser\");\n        \n        if (user != null) {\n            return true;  \u002F\u002F 已登录，放行\n        }\n        \n        \u002F\u002F 未登录\n        \u002F\u002F 方式一：重定向到登录页\n        response.sendRedirect(\"\u002Flogin\");\n        return false;\n        \n        \u002F\u002F 方式二：返回 JSON（API 接口）\n        \u002F\u002F response.setContentType(\"application\u002Fjson;charset=UTF-8\");\n        \u002F\u002F response.getWriter().write(\"{\\\"code\\\":401,\\\"message\\\":\\\"未登录\\\"}\");\n        \u002F\u002F return false;\n    }\n}\n",[983,1869,1870,1876,1891,1895,1901,1915,1923,1935,1940,1954,1974,1978,1991,2003,2007,2011,2016,2021,2036,2046,2050,2055,2060,2065,2070,2074],{"__ignoreMap":11},[1000,1871,1872,1874],{"class":1002,"line":1003},[1000,1873,1007],{"class":1006},[1000,1875,1011],{"class":1010},[1000,1877,1878,1880,1882,1885,1887,1889],{"class":1002,"line":1014},[1000,1879,1017],{"class":1010},[1000,1881,1020],{"class":1010},[1000,1883,1884],{"class":1023}," LoginInterceptor",[1000,1886,1027],{"class":1010},[1000,1888,1030],{"class":1023},[1000,1890,1033],{"class":1006},[1000,1892,1893],{"class":1002,"line":1036},[1000,1894,1039],{"class":1006},[1000,1896,1897,1899],{"class":1002,"line":1042},[1000,1898,1076],{"class":1006},[1000,1900,1079],{"class":1010},[1000,1902,1903,1905,1907,1909,1911,1913],{"class":1002,"line":1049},[1000,1904,1085],{"class":1010},[1000,1906,1088],{"class":1010},[1000,1908,1091],{"class":1023},[1000,1910,1094],{"class":1006},[1000,1912,1098],{"class":1097},[1000,1914,1101],{"class":1006},[1000,1916,1917,1919,1921],{"class":1002,"line":1055},[1000,1918,1107],{"class":1006},[1000,1920,1110],{"class":1097},[1000,1922,1101],{"class":1006},[1000,1924,1925,1927,1929,1931,1933],{"class":1002,"line":1067},[1000,1926,1118],{"class":1006},[1000,1928,1121],{"class":1097},[1000,1930,1124],{"class":1006},[1000,1932,1127],{"class":1010},[1000,1934,1130],{"class":1006},[1000,1936,1937],{"class":1002,"line":1073},[1000,1938,1939],{"class":1045},"        \u002F\u002F 获取 Session 中的用户信息\n",[1000,1941,1942,1945,1947,1949,1952],{"class":1002,"line":1082},[1000,1943,1944],{"class":1006},"        HttpSession session ",[1000,1946,1170],{"class":1010},[1000,1948,1173],{"class":1006},[1000,1950,1951],{"class":1023},"getSession",[1000,1953,1179],{"class":1006},[1000,1955,1956,1959,1961,1964,1967,1969,1972],{"class":1002,"line":1104},[1000,1957,1958],{"class":1006},"        Object user ",[1000,1960,1170],{"class":1010},[1000,1962,1963],{"class":1006}," session.",[1000,1965,1966],{"class":1023},"getAttribute",[1000,1968,1142],{"class":1006},[1000,1970,1971],{"class":1145},"\"loginUser\"",[1000,1973,1149],{"class":1006},[1000,1975,1976],{"class":1002,"line":1115},[1000,1977,1155],{"class":1006},[1000,1979,1980,1982,1985,1987,1989],{"class":1002,"line":1133},[1000,1981,1338],{"class":1010},[1000,1983,1984],{"class":1006}," (user ",[1000,1986,1344],{"class":1010},[1000,1988,1347],{"class":1214},[1000,1990,1350],{"class":1006},[1000,1992,1993,1996,1998,2000],{"class":1002,"line":1152},[1000,1994,1995],{"class":1010},"            return",[1000,1997,1215],{"class":1214},[1000,1999,1218],{"class":1006},[1000,2001,2002],{"class":1045},"\u002F\u002F 已登录，放行\n",[1000,2004,2005],{"class":1002,"line":1158},[1000,2006,1378],{"class":1006},[1000,2008,2009],{"class":1002,"line":1164},[1000,2010,1155],{"class":1006},[1000,2012,2013],{"class":1002,"line":1182},[1000,2014,2015],{"class":1045},"        \u002F\u002F 未登录\n",[1000,2017,2018],{"class":1002,"line":1197},[1000,2019,2020],{"class":1045},"        \u002F\u002F 方式一：重定向到登录页\n",[1000,2022,2023,2026,2029,2031,2034],{"class":1002,"line":1202},[1000,2024,2025],{"class":1006},"        response.",[1000,2027,2028],{"class":1023},"sendRedirect",[1000,2030,1142],{"class":1006},[1000,2032,2033],{"class":1145},"\"\u002Flogin\"",[1000,2035,1149],{"class":1006},[1000,2037,2038,2040,2043],{"class":1002,"line":1208},[1000,2039,1211],{"class":1010},[1000,2041,2042],{"class":1214}," false",[1000,2044,2045],{"class":1006},";\n",[1000,2047,2048],{"class":1002,"line":1224},[1000,2049,1155],{"class":1006},[1000,2051,2052],{"class":1002,"line":1230},[1000,2053,2054],{"class":1045},"        \u002F\u002F 方式二：返回 JSON（API 接口）\n",[1000,2056,2057],{"class":1002,"line":1235},[1000,2058,2059],{"class":1045},"        \u002F\u002F response.setContentType(\"application\u002Fjson;charset=UTF-8\");\n",[1000,2061,2062],{"class":1002,"line":1240},[1000,2063,2064],{"class":1045},"        \u002F\u002F response.getWriter().write(\"{\\\"code\\\":401,\\\"message\\\":\\\"未登录\\\"}\");\n",[1000,2066,2067],{"class":1002,"line":1246},[1000,2068,2069],{"class":1045},"        \u002F\u002F return false;\n",[1000,2071,2072],{"class":1002,"line":1251},[1000,2073,1227],{"class":1006},[1000,2075,2076],{"class":1002,"line":1258},[1000,2077,1498],{"class":1006},[900,2079,2081],{"id":2080},"token-验证拦截器","Token 验证拦截器",[976,2083,2085],{"className":994,"code":2084,"language":996,"meta":11,"style":11},"@Component\npublic class TokenInterceptor implements HandlerInterceptor {\n    \n    @Autowired\n    private TokenService tokenService;\n    \n    @Override\n    public boolean preHandle(HttpServletRequest request, \n                            HttpServletResponse response, \n                            Object handler) throws Exception {\n        \u002F\u002F 获取 Token\n        String token = request.getHeader(\"Authorization\");\n        \n        if (token == null || token.isEmpty()) {\n            sendError(response, 401, \"缺少 Token\");\n            return false;\n        }\n        \n        \u002F\u002F 验证 Token\n        if (!tokenService.verify(token)) {\n            sendError(response, 401, \"Token 无效或已过期\");\n            return false;\n        }\n        \n        \u002F\u002F 将用户信息放入请求属性\n        Long userId = tokenService.getUserId(token);\n        request.setAttribute(\"userId\", userId);\n        \n        return true;\n    }\n    \n    private void sendError(HttpServletResponse response, int code, String message) \n            throws IOException {\n        response.setContentType(\"application\u002Fjson;charset=UTF-8\");\n        response.setStatus(code);\n        response.getWriter().write(\n            String.format(\"{\\\"code\\\":%d,\\\"message\\\":\\\"%s\\\"}\", code, message)\n        );\n    }\n}\n",[983,2086,2087,2093,2108,2112,2118,2125,2129,2135,2149,2157,2169,2174,2193,2197,2221,2239,2247,2251,2255,2260,2279,2294,2302,2306,2310,2315,2331,2347,2351,2359,2363,2367,2398,2406,2420,2430,2446,2491,2496,2500],{"__ignoreMap":11},[1000,2088,2089,2091],{"class":1002,"line":1003},[1000,2090,1007],{"class":1006},[1000,2092,1011],{"class":1010},[1000,2094,2095,2097,2099,2102,2104,2106],{"class":1002,"line":1014},[1000,2096,1017],{"class":1010},[1000,2098,1020],{"class":1010},[1000,2100,2101],{"class":1023}," TokenInterceptor",[1000,2103,1027],{"class":1010},[1000,2105,1030],{"class":1023},[1000,2107,1033],{"class":1006},[1000,2109,2110],{"class":1002,"line":1036},[1000,2111,1039],{"class":1006},[1000,2113,2114,2116],{"class":1002,"line":1042},[1000,2115,1076],{"class":1006},[1000,2117,1675],{"class":1010},[1000,2119,2120,2122],{"class":1002,"line":1049},[1000,2121,1680],{"class":1010},[1000,2123,2124],{"class":1006}," TokenService tokenService;\n",[1000,2126,2127],{"class":1002,"line":1055},[1000,2128,1039],{"class":1006},[1000,2130,2131,2133],{"class":1002,"line":1067},[1000,2132,1076],{"class":1006},[1000,2134,1079],{"class":1010},[1000,2136,2137,2139,2141,2143,2145,2147],{"class":1002,"line":1073},[1000,2138,1085],{"class":1010},[1000,2140,1088],{"class":1010},[1000,2142,1091],{"class":1023},[1000,2144,1094],{"class":1006},[1000,2146,1098],{"class":1097},[1000,2148,1101],{"class":1006},[1000,2150,2151,2153,2155],{"class":1002,"line":1082},[1000,2152,1107],{"class":1006},[1000,2154,1110],{"class":1097},[1000,2156,1101],{"class":1006},[1000,2158,2159,2161,2163,2165,2167],{"class":1002,"line":1104},[1000,2160,1118],{"class":1006},[1000,2162,1121],{"class":1097},[1000,2164,1124],{"class":1006},[1000,2166,1127],{"class":1010},[1000,2168,1130],{"class":1006},[1000,2170,2171],{"class":1002,"line":1115},[1000,2172,2173],{"class":1045},"        \u002F\u002F 获取 Token\n",[1000,2175,2176,2179,2181,2183,2186,2188,2191],{"class":1002,"line":1133},[1000,2177,2178],{"class":1006},"        String token ",[1000,2180,1170],{"class":1010},[1000,2182,1173],{"class":1006},[1000,2184,2185],{"class":1023},"getHeader",[1000,2187,1142],{"class":1006},[1000,2189,2190],{"class":1145},"\"Authorization\"",[1000,2192,1149],{"class":1006},[1000,2194,2195],{"class":1002,"line":1152},[1000,2196,1155],{"class":1006},[1000,2198,2199,2201,2204,2207,2209,2212,2215,2218],{"class":1002,"line":1158},[1000,2200,1338],{"class":1010},[1000,2202,2203],{"class":1006}," (token ",[1000,2205,2206],{"class":1010},"==",[1000,2208,1347],{"class":1214},[1000,2210,2211],{"class":1010}," ||",[1000,2213,2214],{"class":1006}," token.",[1000,2216,2217],{"class":1023},"isEmpty",[1000,2219,2220],{"class":1006},"()) {\n",[1000,2222,2223,2226,2229,2232,2234,2237],{"class":1002,"line":1164},[1000,2224,2225],{"class":1023},"            sendError",[1000,2227,2228],{"class":1006},"(response, ",[1000,2230,2231],{"class":1214},"401",[1000,2233,1367],{"class":1006},[1000,2235,2236],{"class":1145},"\"缺少 Token\"",[1000,2238,1149],{"class":1006},[1000,2240,2241,2243,2245],{"class":1002,"line":1182},[1000,2242,1995],{"class":1010},[1000,2244,2042],{"class":1214},[1000,2246,2045],{"class":1006},[1000,2248,2249],{"class":1002,"line":1197},[1000,2250,1378],{"class":1006},[1000,2252,2253],{"class":1002,"line":1202},[1000,2254,1155],{"class":1006},[1000,2256,2257],{"class":1002,"line":1208},[1000,2258,2259],{"class":1045},"        \u002F\u002F 验证 Token\n",[1000,2261,2262,2264,2267,2270,2273,2276],{"class":1002,"line":1224},[1000,2263,1338],{"class":1010},[1000,2265,2266],{"class":1006}," (",[1000,2268,2269],{"class":1010},"!",[1000,2271,2272],{"class":1006},"tokenService.",[1000,2274,2275],{"class":1023},"verify",[1000,2277,2278],{"class":1006},"(token)) {\n",[1000,2280,2281,2283,2285,2287,2289,2292],{"class":1002,"line":1230},[1000,2282,2225],{"class":1023},[1000,2284,2228],{"class":1006},[1000,2286,2231],{"class":1214},[1000,2288,1367],{"class":1006},[1000,2290,2291],{"class":1145},"\"Token 无效或已过期\"",[1000,2293,1149],{"class":1006},[1000,2295,2296,2298,2300],{"class":1002,"line":1235},[1000,2297,1995],{"class":1010},[1000,2299,2042],{"class":1214},[1000,2301,2045],{"class":1006},[1000,2303,2304],{"class":1002,"line":1240},[1000,2305,1378],{"class":1006},[1000,2307,2308],{"class":1002,"line":1246},[1000,2309,1155],{"class":1006},[1000,2311,2312],{"class":1002,"line":1251},[1000,2313,2314],{"class":1045},"        \u002F\u002F 将用户信息放入请求属性\n",[1000,2316,2317,2320,2322,2325,2328],{"class":1002,"line":1258},[1000,2318,2319],{"class":1006},"        Long userId ",[1000,2321,1170],{"class":1010},[1000,2323,2324],{"class":1006}," tokenService.",[1000,2326,2327],{"class":1023},"getUserId",[1000,2329,2330],{"class":1006},"(token);\n",[1000,2332,2333,2336,2339,2341,2344],{"class":1002,"line":1275},[1000,2334,2335],{"class":1006},"        request.",[1000,2337,2338],{"class":1023},"setAttribute",[1000,2340,1142],{"class":1006},[1000,2342,2343],{"class":1145},"\"userId\"",[1000,2345,2346],{"class":1006},", userId);\n",[1000,2348,2349],{"class":1002,"line":1285},[1000,2350,1155],{"class":1006},[1000,2352,2353,2355,2357],{"class":1002,"line":1295},[1000,2354,1211],{"class":1010},[1000,2356,1215],{"class":1214},[1000,2358,2045],{"class":1006},[1000,2360,2361],{"class":1002,"line":1310},[1000,2362,1227],{"class":1006},[1000,2364,2365],{"class":1002,"line":1324},[1000,2366,1039],{"class":1006},[1000,2368,2369,2371,2373,2376,2379,2381,2383,2386,2389,2392,2395],{"class":1002,"line":1329},[1000,2370,1680],{"class":1010},[1000,2372,1263],{"class":1010},[1000,2374,2375],{"class":1023}," sendError",[1000,2377,2378],{"class":1006},"(HttpServletResponse ",[1000,2380,1110],{"class":1097},[1000,2382,1367],{"class":1006},[1000,2384,2385],{"class":1010},"int",[1000,2387,2388],{"class":1097}," code",[1000,2390,2391],{"class":1006},", String ",[1000,2393,2394],{"class":1097},"message",[1000,2396,2397],{"class":1006},") \n",[1000,2399,2400,2403],{"class":1002,"line":1335},[1000,2401,2402],{"class":1010},"            throws",[1000,2404,2405],{"class":1006}," IOException {\n",[1000,2407,2408,2410,2413,2415,2418],{"class":1002,"line":1353},[1000,2409,2025],{"class":1006},[1000,2411,2412],{"class":1023},"setContentType",[1000,2414,1142],{"class":1006},[1000,2416,2417],{"class":1145},"\"application\u002Fjson;charset=UTF-8\"",[1000,2419,1149],{"class":1006},[1000,2421,2422,2424,2427],{"class":1002,"line":1375},[1000,2423,2025],{"class":1006},[1000,2425,2426],{"class":1023},"setStatus",[1000,2428,2429],{"class":1006},"(code);\n",[1000,2431,2432,2434,2437,2440,2443],{"class":1002,"line":1381},[1000,2433,2025],{"class":1006},[1000,2435,2436],{"class":1023},"getWriter",[1000,2438,2439],{"class":1006},"().",[1000,2441,2442],{"class":1023},"write",[1000,2444,2445],{"class":1006},"(\n",[1000,2447,2448,2451,2454,2456,2459,2462,2464,2466,2469,2471,2473,2475,2478,2480,2483,2485,2488],{"class":1002,"line":1386},[1000,2449,2450],{"class":1006},"            String.",[1000,2452,2453],{"class":1023},"format",[1000,2455,1142],{"class":1006},[1000,2457,2458],{"class":1145},"\"{",[1000,2460,2461],{"class":1214},"\\\"",[1000,2463,983],{"class":1145},[1000,2465,2461],{"class":1214},[1000,2467,2468],{"class":1145},":%d,",[1000,2470,2461],{"class":1214},[1000,2472,2394],{"class":1145},[1000,2474,2461],{"class":1214},[1000,2476,2477],{"class":1145},":",[1000,2479,2461],{"class":1214},[1000,2481,2482],{"class":1145},"%s",[1000,2484,2461],{"class":1214},[1000,2486,2487],{"class":1145},"}\"",[1000,2489,2490],{"class":1006},", code, message)\n",[1000,2492,2493],{"class":1002,"line":1391},[1000,2494,2495],{"class":1006},"        );\n",[1000,2497,2498],{"class":1002,"line":1396},[1000,2499,1227],{"class":1006},[1000,2501,2502],{"class":1002,"line":1402},[1000,2503,1498],{"class":1006},[900,2505,2506],{"id":2506},"日志拦截器",[976,2508,2510],{"className":994,"code":2509,"language":996,"meta":11,"style":11},"@Component\n@Slf4j\npublic class LogInterceptor implements HandlerInterceptor {\n    \n    private static final String START_TIME = \"requestStartTime\";\n    \n    @Override\n    public boolean preHandle(HttpServletRequest request, \n                            HttpServletResponse response, \n                            Object handler) throws Exception {\n        \u002F\u002F 记录开始时间\n        request.setAttribute(START_TIME, System.currentTimeMillis());\n        \n        \u002F\u002F 记录请求信息\n        log.info(\"请求开始 - URI: {}, Method: {}, IP: {}\", \n                request.getRequestURI(),\n                request.getMethod(),\n                getClientIP(request));\n        \n        return true;\n    }\n    \n    @Override\n    public void afterCompletion(HttpServletRequest request, \n                               HttpServletResponse response, \n                               Object handler, \n                               Exception ex) throws Exception {\n        \u002F\u002F 计算耗时\n        Long startTime = (Long) request.getAttribute(START_TIME);\n        long duration = System.currentTimeMillis() - startTime;\n        \n        \u002F\u002F 记录响应信息\n        log.info(\"请求结束 - URI: {}, Status: {}, 耗时: {}ms\", \n                request.getRequestURI(),\n                response.getStatus(),\n                duration);\n        \n        \u002F\u002F 记录异常\n        if (ex != null) {\n            log.error(\"请求异常 - URI: {}, Error: {}\", \n                    request.getRequestURI(), ex.getMessage());\n        }\n    }\n    \n    private String getClientIP(HttpServletRequest request) {\n        String ip = request.getHeader(\"X-Forwarded-For\");\n        if (ip == null || ip.isEmpty() || \"unknown\".equalsIgnoreCase(ip)) {\n            ip = request.getHeader(\"X-Real-IP\");\n        }\n        if (ip == null || ip.isEmpty() || \"unknown\".equalsIgnoreCase(ip)) {\n            ip = request.getRemoteAddr();\n        }\n        return ip;\n    }\n}\n",[983,2511,2512,2518,2525,2540,2544,2564,2568,2574,2588,2596,2608,2613,2628,2632,2637,2652,2662,2670,2678,2682,2690,2694,2698,2704,2718,2726,2734,2746,2751,2766,2790,2794,2799,2812,2820,2830,2835,2839,2844,2857,2872,2887,2891,2895,2899,2915,2933,2968,2986,2990,3018,3032,3037,3045,3050],{"__ignoreMap":11},[1000,2513,2514,2516],{"class":1002,"line":1003},[1000,2515,1007],{"class":1006},[1000,2517,1011],{"class":1010},[1000,2519,2520,2522],{"class":1002,"line":1014},[1000,2521,1007],{"class":1006},[1000,2523,2524],{"class":1010},"Slf4j\n",[1000,2526,2527,2529,2531,2534,2536,2538],{"class":1002,"line":1036},[1000,2528,1017],{"class":1010},[1000,2530,1020],{"class":1010},[1000,2532,2533],{"class":1023}," LogInterceptor",[1000,2535,1027],{"class":1010},[1000,2537,1030],{"class":1023},[1000,2539,1033],{"class":1006},[1000,2541,2542],{"class":1002,"line":1042},[1000,2543,1039],{"class":1006},[1000,2545,2546,2548,2551,2554,2557,2559,2562],{"class":1002,"line":1049},[1000,2547,1680],{"class":1010},[1000,2549,2550],{"class":1010}," static",[1000,2552,2553],{"class":1010}," final",[1000,2555,2556],{"class":1006}," String START_TIME ",[1000,2558,1170],{"class":1010},[1000,2560,2561],{"class":1145}," \"requestStartTime\"",[1000,2563,2045],{"class":1006},[1000,2565,2566],{"class":1002,"line":1055},[1000,2567,1039],{"class":1006},[1000,2569,2570,2572],{"class":1002,"line":1067},[1000,2571,1076],{"class":1006},[1000,2573,1079],{"class":1010},[1000,2575,2576,2578,2580,2582,2584,2586],{"class":1002,"line":1073},[1000,2577,1085],{"class":1010},[1000,2579,1088],{"class":1010},[1000,2581,1091],{"class":1023},[1000,2583,1094],{"class":1006},[1000,2585,1098],{"class":1097},[1000,2587,1101],{"class":1006},[1000,2589,2590,2592,2594],{"class":1002,"line":1082},[1000,2591,1107],{"class":1006},[1000,2593,1110],{"class":1097},[1000,2595,1101],{"class":1006},[1000,2597,2598,2600,2602,2604,2606],{"class":1002,"line":1104},[1000,2599,1118],{"class":1006},[1000,2601,1121],{"class":1097},[1000,2603,1124],{"class":1006},[1000,2605,1127],{"class":1010},[1000,2607,1130],{"class":1006},[1000,2609,2610],{"class":1002,"line":1115},[1000,2611,2612],{"class":1045},"        \u002F\u002F 记录开始时间\n",[1000,2614,2615,2617,2619,2622,2625],{"class":1002,"line":1133},[1000,2616,2335],{"class":1006},[1000,2618,2338],{"class":1023},[1000,2620,2621],{"class":1006},"(START_TIME, System.",[1000,2623,2624],{"class":1023},"currentTimeMillis",[1000,2626,2627],{"class":1006},"());\n",[1000,2629,2630],{"class":1002,"line":1152},[1000,2631,1155],{"class":1006},[1000,2633,2634],{"class":1002,"line":1158},[1000,2635,2636],{"class":1045},"        \u002F\u002F 记录请求信息\n",[1000,2638,2639,2642,2645,2647,2650],{"class":1002,"line":1164},[1000,2640,2641],{"class":1006},"        log.",[1000,2643,2644],{"class":1023},"info",[1000,2646,1142],{"class":1006},[1000,2648,2649],{"class":1145},"\"请求开始 - URI: {}, Method: {}, IP: {}\"",[1000,2651,1101],{"class":1006},[1000,2653,2654,2657,2659],{"class":1002,"line":1182},[1000,2655,2656],{"class":1006},"                request.",[1000,2658,1176],{"class":1023},[1000,2660,2661],{"class":1006},"(),\n",[1000,2663,2664,2666,2668],{"class":1002,"line":1197},[1000,2665,2656],{"class":1006},[1000,2667,1192],{"class":1023},[1000,2669,2661],{"class":1006},[1000,2671,2672,2675],{"class":1002,"line":1202},[1000,2673,2674],{"class":1023},"                getClientIP",[1000,2676,2677],{"class":1006},"(request));\n",[1000,2679,2680],{"class":1002,"line":1208},[1000,2681,1155],{"class":1006},[1000,2683,2684,2686,2688],{"class":1002,"line":1224},[1000,2685,1211],{"class":1010},[1000,2687,1215],{"class":1214},[1000,2689,2045],{"class":1006},[1000,2691,2692],{"class":1002,"line":1230},[1000,2693,1227],{"class":1006},[1000,2695,2696],{"class":1002,"line":1235},[1000,2697,1039],{"class":1006},[1000,2699,2700,2702],{"class":1002,"line":1240},[1000,2701,1076],{"class":1006},[1000,2703,1079],{"class":1010},[1000,2705,2706,2708,2710,2712,2714,2716],{"class":1002,"line":1246},[1000,2707,1085],{"class":1010},[1000,2709,1263],{"class":1010},[1000,2711,1421],{"class":1023},[1000,2713,1094],{"class":1006},[1000,2715,1098],{"class":1097},[1000,2717,1101],{"class":1006},[1000,2719,2720,2722,2724],{"class":1002,"line":1251},[1000,2721,1433],{"class":1006},[1000,2723,1110],{"class":1097},[1000,2725,1101],{"class":1006},[1000,2727,2728,2730,2732],{"class":1002,"line":1258},[1000,2729,1443],{"class":1006},[1000,2731,1121],{"class":1097},[1000,2733,1101],{"class":1006},[1000,2735,2736,2738,2740,2742,2744],{"class":1002,"line":1275},[1000,2737,1453],{"class":1006},[1000,2739,1456],{"class":1097},[1000,2741,1124],{"class":1006},[1000,2743,1127],{"class":1010},[1000,2745,1130],{"class":1006},[1000,2747,2748],{"class":1002,"line":1285},[1000,2749,2750],{"class":1045},"        \u002F\u002F 计算耗时\n",[1000,2752,2753,2756,2758,2761,2763],{"class":1002,"line":1295},[1000,2754,2755],{"class":1006},"        Long startTime ",[1000,2757,1170],{"class":1010},[1000,2759,2760],{"class":1006}," (Long) request.",[1000,2762,1966],{"class":1023},[1000,2764,2765],{"class":1006},"(START_TIME);\n",[1000,2767,2768,2771,2774,2776,2779,2781,2784,2787],{"class":1002,"line":1310},[1000,2769,2770],{"class":1010},"        long",[1000,2772,2773],{"class":1006}," duration ",[1000,2775,1170],{"class":1010},[1000,2777,2778],{"class":1006}," System.",[1000,2780,2624],{"class":1023},[1000,2782,2783],{"class":1006},"() ",[1000,2785,2786],{"class":1010},"-",[1000,2788,2789],{"class":1006}," startTime;\n",[1000,2791,2792],{"class":1002,"line":1324},[1000,2793,1155],{"class":1006},[1000,2795,2796],{"class":1002,"line":1329},[1000,2797,2798],{"class":1045},"        \u002F\u002F 记录响应信息\n",[1000,2800,2801,2803,2805,2807,2810],{"class":1002,"line":1335},[1000,2802,2641],{"class":1006},[1000,2804,2644],{"class":1023},[1000,2806,1142],{"class":1006},[1000,2808,2809],{"class":1145},"\"请求结束 - URI: {}, Status: {}, 耗时: {}ms\"",[1000,2811,1101],{"class":1006},[1000,2813,2814,2816,2818],{"class":1002,"line":1353},[1000,2815,2656],{"class":1006},[1000,2817,1176],{"class":1023},[1000,2819,2661],{"class":1006},[1000,2821,2822,2825,2828],{"class":1002,"line":1375},[1000,2823,2824],{"class":1006},"                response.",[1000,2826,2827],{"class":1023},"getStatus",[1000,2829,2661],{"class":1006},[1000,2831,2832],{"class":1002,"line":1381},[1000,2833,2834],{"class":1006},"                duration);\n",[1000,2836,2837],{"class":1002,"line":1386},[1000,2838,1155],{"class":1006},[1000,2840,2841],{"class":1002,"line":1391},[1000,2842,2843],{"class":1045},"        \u002F\u002F 记录异常\n",[1000,2845,2846,2848,2851,2853,2855],{"class":1002,"line":1396},[1000,2847,1338],{"class":1010},[1000,2849,2850],{"class":1006}," (ex ",[1000,2852,1344],{"class":1010},[1000,2854,1347],{"class":1214},[1000,2856,1350],{"class":1006},[1000,2858,2859,2862,2865,2867,2870],{"class":1002,"line":1402},[1000,2860,2861],{"class":1006},"            log.",[1000,2863,2864],{"class":1023},"error",[1000,2866,1142],{"class":1006},[1000,2868,2869],{"class":1145},"\"请求异常 - URI: {}, Error: {}\"",[1000,2871,1101],{"class":1006},[1000,2873,2874,2877,2879,2882,2885],{"class":1002,"line":1407},[1000,2875,2876],{"class":1006},"                    request.",[1000,2878,1176],{"class":1023},[1000,2880,2881],{"class":1006},"(), ex.",[1000,2883,2884],{"class":1023},"getMessage",[1000,2886,2627],{"class":1006},[1000,2888,2889],{"class":1002,"line":1414},[1000,2890,1378],{"class":1006},[1000,2892,2893],{"class":1002,"line":1430},[1000,2894,1227],{"class":1006},[1000,2896,2897],{"class":1002,"line":1440},[1000,2898,1039],{"class":1006},[1000,2900,2901,2903,2906,2909,2911,2913],{"class":1002,"line":1450},[1000,2902,1680],{"class":1010},[1000,2904,2905],{"class":1006}," String ",[1000,2907,2908],{"class":1023},"getClientIP",[1000,2910,1094],{"class":1006},[1000,2912,1098],{"class":1097},[1000,2914,1350],{"class":1006},[1000,2916,2917,2920,2922,2924,2926,2928,2931],{"class":1002,"line":1465},[1000,2918,2919],{"class":1006},"        String ip ",[1000,2921,1170],{"class":1010},[1000,2923,1173],{"class":1006},[1000,2925,2185],{"class":1023},[1000,2927,1142],{"class":1006},[1000,2929,2930],{"class":1145},"\"X-Forwarded-For\"",[1000,2932,1149],{"class":1006},[1000,2934,2935,2937,2940,2942,2944,2946,2949,2951,2953,2956,2959,2962,2965],{"class":1002,"line":1479},[1000,2936,1338],{"class":1010},[1000,2938,2939],{"class":1006}," (ip ",[1000,2941,2206],{"class":1010},[1000,2943,1347],{"class":1214},[1000,2945,2211],{"class":1010},[1000,2947,2948],{"class":1006}," ip.",[1000,2950,2217],{"class":1023},[1000,2952,2783],{"class":1006},[1000,2954,2955],{"class":1010},"||",[1000,2957,2958],{"class":1145}," \"unknown\"",[1000,2960,2961],{"class":1006},".",[1000,2963,2964],{"class":1023},"equalsIgnoreCase",[1000,2966,2967],{"class":1006},"(ip)) {\n",[1000,2969,2970,2973,2975,2977,2979,2981,2984],{"class":1002,"line":1484},[1000,2971,2972],{"class":1006},"            ip ",[1000,2974,1170],{"class":1010},[1000,2976,1173],{"class":1006},[1000,2978,2185],{"class":1023},[1000,2980,1142],{"class":1006},[1000,2982,2983],{"class":1145},"\"X-Real-IP\"",[1000,2985,1149],{"class":1006},[1000,2987,2988],{"class":1002,"line":1490},[1000,2989,1378],{"class":1006},[1000,2991,2992,2994,2996,2998,3000,3002,3004,3006,3008,3010,3012,3014,3016],{"class":1002,"line":1495},[1000,2993,1338],{"class":1010},[1000,2995,2939],{"class":1006},[1000,2997,2206],{"class":1010},[1000,2999,1347],{"class":1214},[1000,3001,2211],{"class":1010},[1000,3003,2948],{"class":1006},[1000,3005,2217],{"class":1023},[1000,3007,2783],{"class":1006},[1000,3009,2955],{"class":1010},[1000,3011,2958],{"class":1145},[1000,3013,2961],{"class":1006},[1000,3015,2964],{"class":1023},[1000,3017,2967],{"class":1006},[1000,3019,3021,3023,3025,3027,3030],{"class":1002,"line":3020},51,[1000,3022,2972],{"class":1006},[1000,3024,1170],{"class":1010},[1000,3026,1173],{"class":1006},[1000,3028,3029],{"class":1023},"getRemoteAddr",[1000,3031,1179],{"class":1006},[1000,3033,3035],{"class":1002,"line":3034},52,[1000,3036,1378],{"class":1006},[1000,3038,3040,3042],{"class":1002,"line":3039},53,[1000,3041,1211],{"class":1010},[1000,3043,3044],{"class":1006}," ip;\n",[1000,3046,3048],{"class":1002,"line":3047},54,[1000,3049,1227],{"class":1006},[1000,3051,3053],{"class":1002,"line":3052},55,[1000,3054,1498],{"class":1006},[900,3056,3057],{"id":3057},"跨域拦截器",[976,3059,3061],{"className":994,"code":3060,"language":996,"meta":11,"style":11},"@Component\npublic class CorsInterceptor implements HandlerInterceptor {\n    \n    @Override\n    public boolean preHandle(HttpServletRequest request, \n                            HttpServletResponse response, \n                            Object handler) throws Exception {\n        \u002F\u002F 设置跨域响应头\n        response.setHeader(\"Access-Control-Allow-Origin\", \"*\");\n        response.setHeader(\"Access-Control-Allow-Methods\", \"GET, POST, PUT, DELETE, OPTIONS\");\n        response.setHeader(\"Access-Control-Allow-Headers\", \"Content-Type, Authorization\");\n        response.setHeader(\"Access-Control-Max-Age\", \"3600\");\n        \n        \u002F\u002F 处理预检请求\n        if (\"OPTIONS\".equalsIgnoreCase(request.getMethod())) {\n            response.setStatus(HttpServletResponse.SC_OK);\n            return false;\n        }\n        \n        return true;\n    }\n}\n",[983,3062,3063,3069,3084,3088,3094,3108,3116,3128,3133,3152,3170,3188,3206,3210,3215,3236,3246,3254,3258,3262,3270,3274],{"__ignoreMap":11},[1000,3064,3065,3067],{"class":1002,"line":1003},[1000,3066,1007],{"class":1006},[1000,3068,1011],{"class":1010},[1000,3070,3071,3073,3075,3078,3080,3082],{"class":1002,"line":1014},[1000,3072,1017],{"class":1010},[1000,3074,1020],{"class":1010},[1000,3076,3077],{"class":1023}," CorsInterceptor",[1000,3079,1027],{"class":1010},[1000,3081,1030],{"class":1023},[1000,3083,1033],{"class":1006},[1000,3085,3086],{"class":1002,"line":1036},[1000,3087,1039],{"class":1006},[1000,3089,3090,3092],{"class":1002,"line":1042},[1000,3091,1076],{"class":1006},[1000,3093,1079],{"class":1010},[1000,3095,3096,3098,3100,3102,3104,3106],{"class":1002,"line":1049},[1000,3097,1085],{"class":1010},[1000,3099,1088],{"class":1010},[1000,3101,1091],{"class":1023},[1000,3103,1094],{"class":1006},[1000,3105,1098],{"class":1097},[1000,3107,1101],{"class":1006},[1000,3109,3110,3112,3114],{"class":1002,"line":1055},[1000,3111,1107],{"class":1006},[1000,3113,1110],{"class":1097},[1000,3115,1101],{"class":1006},[1000,3117,3118,3120,3122,3124,3126],{"class":1002,"line":1067},[1000,3119,1118],{"class":1006},[1000,3121,1121],{"class":1097},[1000,3123,1124],{"class":1006},[1000,3125,1127],{"class":1010},[1000,3127,1130],{"class":1006},[1000,3129,3130],{"class":1002,"line":1073},[1000,3131,3132],{"class":1045},"        \u002F\u002F 设置跨域响应头\n",[1000,3134,3135,3137,3140,3142,3145,3147,3150],{"class":1002,"line":1082},[1000,3136,2025],{"class":1006},[1000,3138,3139],{"class":1023},"setHeader",[1000,3141,1142],{"class":1006},[1000,3143,3144],{"class":1145},"\"Access-Control-Allow-Origin\"",[1000,3146,1367],{"class":1006},[1000,3148,3149],{"class":1145},"\"*\"",[1000,3151,1149],{"class":1006},[1000,3153,3154,3156,3158,3160,3163,3165,3168],{"class":1002,"line":1104},[1000,3155,2025],{"class":1006},[1000,3157,3139],{"class":1023},[1000,3159,1142],{"class":1006},[1000,3161,3162],{"class":1145},"\"Access-Control-Allow-Methods\"",[1000,3164,1367],{"class":1006},[1000,3166,3167],{"class":1145},"\"GET, POST, PUT, DELETE, OPTIONS\"",[1000,3169,1149],{"class":1006},[1000,3171,3172,3174,3176,3178,3181,3183,3186],{"class":1002,"line":1115},[1000,3173,2025],{"class":1006},[1000,3175,3139],{"class":1023},[1000,3177,1142],{"class":1006},[1000,3179,3180],{"class":1145},"\"Access-Control-Allow-Headers\"",[1000,3182,1367],{"class":1006},[1000,3184,3185],{"class":1145},"\"Content-Type, Authorization\"",[1000,3187,1149],{"class":1006},[1000,3189,3190,3192,3194,3196,3199,3201,3204],{"class":1002,"line":1133},[1000,3191,2025],{"class":1006},[1000,3193,3139],{"class":1023},[1000,3195,1142],{"class":1006},[1000,3197,3198],{"class":1145},"\"Access-Control-Max-Age\"",[1000,3200,1367],{"class":1006},[1000,3202,3203],{"class":1145},"\"3600\"",[1000,3205,1149],{"class":1006},[1000,3207,3208],{"class":1002,"line":1152},[1000,3209,1155],{"class":1006},[1000,3211,3212],{"class":1002,"line":1158},[1000,3213,3214],{"class":1045},"        \u002F\u002F 处理预检请求\n",[1000,3216,3217,3219,3221,3224,3226,3228,3231,3233],{"class":1002,"line":1164},[1000,3218,1338],{"class":1010},[1000,3220,2266],{"class":1006},[1000,3222,3223],{"class":1145},"\"OPTIONS\"",[1000,3225,2961],{"class":1006},[1000,3227,2964],{"class":1023},[1000,3229,3230],{"class":1006},"(request.",[1000,3232,1192],{"class":1023},[1000,3234,3235],{"class":1006},"())) {\n",[1000,3237,3238,3241,3243],{"class":1002,"line":1182},[1000,3239,3240],{"class":1006},"            response.",[1000,3242,2426],{"class":1023},[1000,3244,3245],{"class":1006},"(HttpServletResponse.SC_OK);\n",[1000,3247,3248,3250,3252],{"class":1002,"line":1197},[1000,3249,1995],{"class":1010},[1000,3251,2042],{"class":1214},[1000,3253,2045],{"class":1006},[1000,3255,3256],{"class":1002,"line":1202},[1000,3257,1378],{"class":1006},[1000,3259,3260],{"class":1002,"line":1208},[1000,3261,1155],{"class":1006},[1000,3263,3264,3266,3268],{"class":1002,"line":1224},[1000,3265,1211],{"class":1010},[1000,3267,1215],{"class":1214},[1000,3269,2045],{"class":1006},[1000,3271,3272],{"class":1002,"line":1230},[1000,3273,1227],{"class":1006},[1000,3275,3276],{"class":1002,"line":1235},[1000,3277,1498],{"class":1006},[892,3279,3280],{"id":3280},"多个拦截器执行顺序",[896,3282,3283],{},"当有多个拦截器时，执行顺序如下：",[976,3285,3288],{"className":3286,"code":3287,"language":981},[979],"preHandle1 → preHandle2 → preHandle3\n                  ↓\n            Controller\n                  ↓\npostHandle3 → postHandle2 → postHandle1\n                  ↓\n              视图渲染\n                  ↓\nafterCompletion3 → afterCompletion2 → afterCompletion1\n",[983,3289,3287],{"__ignoreMap":11},[3291,3292,3293,3297,3300],"ul",{},[3294,3295,3296],"li",{},"preHandle：按注册顺序执行",[3294,3298,3299],{},"postHandle：按注册逆序执行",[3294,3301,3302],{},"afterCompletion：按注册逆序执行",[3304,3305,3307],"warning",{"title":3306},"注意","\n如果某个拦截器的 preHandle 返回 false，后续拦截器的 preHandle 不会执行，但之前已执行的拦截器的 afterCompletion 会执行。\n",[3309,3310,3311],"style",{},"html pre.shiki code .sxrX7, html code.shiki .sxrX7{--shiki-light:#24292E;--shiki-default:#24292E;--shiki-dark:#E1E4E8}html pre.shiki code .s8jYJ, html code.shiki .s8jYJ{--shiki-light:#D73A49;--shiki-default:#D73A49;--shiki-dark:#F97583}html pre.shiki code .snPdu, html code.shiki .snPdu{--shiki-light:#6F42C1;--shiki-default:#6F42C1;--shiki-dark:#B392F0}html pre.shiki code .sCsY4, html code.shiki .sCsY4{--shiki-light:#6A737D;--shiki-default:#6A737D;--shiki-dark:#6A737D}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}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 .sovSZ, html code.shiki .sovSZ{--shiki-light:#22863A;--shiki-default:#22863A;--shiki-dark:#85E89D}",{"title":11,"searchDepth":1014,"depth":1014,"links":3313},[3314,3318,3321,3325,3331],{"id":894,"depth":1014,"text":894,"children":3315},[3316,3317],{"id":902,"depth":1036,"text":903},{"id":974,"depth":1036,"text":974},{"id":987,"depth":1014,"text":987,"children":3319},[3320],{"id":990,"depth":1036,"text":991},{"id":1501,"depth":1014,"text":1501,"children":3322},[3323,3324],{"id":1504,"depth":1036,"text":1505},{"id":1635,"depth":1036,"text":1636},{"id":1861,"depth":1014,"text":1861,"children":3326},[3327,3328,3329,3330],{"id":1864,"depth":1036,"text":1864},{"id":2080,"depth":1036,"text":2081},{"id":2506,"depth":1036,"text":2506},{"id":3057,"depth":1036,"text":3057},{"id":3280,"depth":1014,"text":3280},"md",{},true,{"title":565,"description":11},"other\u002Fspring-series\u002Fspringmvc\u002Fspringmvc-interceptor","AE3cPbhoQY_I2Azc3KSnqu_T3QHkn_KV2tS0k1CN1iw",1775496431407]