[{"data":1,"prerenderedAt":3010},["ShallowReactive",2],{"search-docs":3,"doc-\u002Fother\u002Fjava\u002Fjvm\u002Fclass-loading":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":444,"body":888,"description":445,"extension":3005,"meta":3006,"navigation":1555,"path":443,"seo":3007,"stem":3008,"__hash__":3009},"docs\u002Fother\u002Fjava\u002Fjvm\u002Fclass-loading.md",{"type":889,"value":890,"toc":2961},"minimark",[891,900,904,914,918,921,925,942,945,966,970,973,976,1028,1031,1065,1069,1072,1075,1170,1173,1259,1263,1266,1270,1299,1302,1316,1320,1327,1333,1444,1447,1450,1489,1492,1742,1745,1748,1754,1757,1938,1941,1944,1950,1953,2217,2220,2233,2276,2279,2282,2293,2690,2693,2817,2820,2823,2843,2846,2900,2904,2910,2921,2924,2957],[892,893,894,895,899],"p",{},"类加载机制是 JVM 将 ",[896,897,898],"code",{},".class"," 文件加载到内存，并对数据进行校验、转换解析和初始化，最终形成可被 JVM 直接使用的 Java 类型的过程。",[901,902,903],"h2",{"id":903},"类的生命周期",[905,906,912],"pre",{"className":907,"code":909,"language":910,"meta":911},[908],"language-text","flowchart LR\n    Load[\"加载\u003Cbr\u002F>Loading\"] --> Verify[\"验证\u003Cbr\u002F>Verifying\"]\n    Verify --> Prepare[\"准备\u003Cbr\u002F>Preparing\"]\n    Prepare --> Resolve[\"解析\u003Cbr\u002F>Resolving\"]\n    Resolve --> Init[\"初始化\u003Cbr\u002F>Initializing\"]\n    Init --> Use[\"使用\u003Cbr\u002F>Using\"]\n    Use --> Unload[\"卸载\u003Cbr\u002F>Unloading\"]\n    subgraph Linking[\"连接 #40;Linking#41;\"]\n        Verify\n        Prepare\n        Resolve\n    end\n","text","mermaid",[896,913,909],{"__ignoreMap":11},[901,915,917],{"id":916},"加载loading","加载（Loading）",[892,919,920],{},"加载是类加载的第一个阶段。",[922,923,924],"h3",{"id":924},"加载过程",[926,927,928,932,935],"ol",{},[929,930,931],"li",{},"通过类的全限定名获取定义此类的二进制字节流",[929,933,934],{},"将字节流所代表的静态存储结构转化为方法区的运行时数据结构",[929,936,937,938,941],{},"在内存中生成一个代表这个类的 ",[896,939,940],{},"java.lang.Class"," 对象",[922,943,944],{"id":944},"类的来源",[946,947,948,954,957,960,963],"ul",{},[929,949,950,951,953],{},"本地文件系统的 ",[896,952,898],{}," 文件",[929,955,956],{},"JAR、WAR 包中的类",[929,958,959],{},"网络下载的类",[929,961,962],{},"运行时动态生成的类（动态代理）",[929,964,965],{},"其他文件生成（JSP 编译后的类）",[901,967,969],{"id":968},"验证verification","验证（Verification）",[892,971,972],{},"确保 Class 文件的字节流符合 JVM 规范，不会危害虚拟机。",[922,974,975],{"id":975},"验证阶段",[977,978,979,992],"table",{},[980,981,982],"thead",{},[983,984,985,989],"tr",{},[986,987,988],"th",{},"阶段",[986,990,991],{},"说明",[993,994,995,1004,1012,1020],"tbody",{},[983,996,997,1001],{},[998,999,1000],"td",{},"文件格式验证",[998,1002,1003],{},"魔数、版本号、常量池等",[983,1005,1006,1009],{},[998,1007,1008],{},"元数据验证",[998,1010,1011],{},"语义分析，如是否有父类",[983,1013,1014,1017],{},[998,1015,1016],{},"字节码验证",[998,1018,1019],{},"数据流和控制流分析",[983,1021,1022,1025],{},[998,1023,1024],{},"符号引用验证",[998,1026,1027],{},"符号引用能否找到对应的类、方法、字段",[922,1029,1030],{"id":1030},"魔数验证",[905,1032,1036],{"className":1033,"code":1034,"language":1035,"meta":11,"style":11},"language-java shiki shiki-themes github-light github-light github-dark","\u002F\u002F Class 文件以魔数开头：0xCAFEBABE\n\u002F\u002F 可以使用 xxd 命令查看\n\u002F\u002F $ xxd Test.class | head -1\n\u002F\u002F 00000000: cafe babe 0000 0034 0022 ...\n","java",[896,1037,1038,1047,1053,1059],{"__ignoreMap":11},[1039,1040,1043],"span",{"class":1041,"line":1042},"line",1,[1039,1044,1046],{"class":1045},"sCsY4","\u002F\u002F Class 文件以魔数开头：0xCAFEBABE\n",[1039,1048,1050],{"class":1041,"line":1049},2,[1039,1051,1052],{"class":1045},"\u002F\u002F 可以使用 xxd 命令查看\n",[1039,1054,1056],{"class":1041,"line":1055},3,[1039,1057,1058],{"class":1045},"\u002F\u002F $ xxd Test.class | head -1\n",[1039,1060,1062],{"class":1041,"line":1061},4,[1039,1063,1064],{"class":1045},"\u002F\u002F 00000000: cafe babe 0000 0034 0022 ...\n",[901,1066,1068],{"id":1067},"准备preparation","准备（Preparation）",[892,1070,1071],{},"为类变量（static 变量）分配内存并设置初始值。",[922,1073,1074],{"id":1074},"初始值规则",[905,1076,1078],{"className":1033,"code":1077,"language":1035,"meta":11,"style":11},"public class PrepareDemo {\n    \u002F\u002F 准备阶段：value = 0（不是 123）\n    \u002F\u002F 初始化阶段：value = 123\n    public static int value = 123;\n    \n    \u002F\u002F 准备阶段：直接赋值 123（final 常量）\n    public static final int CONST = 123;\n}\n",[896,1079,1080,1097,1102,1107,1131,1137,1143,1164],{"__ignoreMap":11},[1039,1081,1082,1086,1089,1093],{"class":1041,"line":1042},[1039,1083,1085],{"class":1084},"s8jYJ","public",[1039,1087,1088],{"class":1084}," class",[1039,1090,1092],{"class":1091},"snPdu"," PrepareDemo",[1039,1094,1096],{"class":1095},"sxrX7"," {\n",[1039,1098,1099],{"class":1041,"line":1049},[1039,1100,1101],{"class":1045},"    \u002F\u002F 准备阶段：value = 0（不是 123）\n",[1039,1103,1104],{"class":1041,"line":1055},[1039,1105,1106],{"class":1045},"    \u002F\u002F 初始化阶段：value = 123\n",[1039,1108,1109,1112,1115,1118,1121,1124,1128],{"class":1041,"line":1061},[1039,1110,1111],{"class":1084},"    public",[1039,1113,1114],{"class":1084}," static",[1039,1116,1117],{"class":1084}," int",[1039,1119,1120],{"class":1095}," value ",[1039,1122,1123],{"class":1084},"=",[1039,1125,1127],{"class":1126},"sBjJW"," 123",[1039,1129,1130],{"class":1095},";\n",[1039,1132,1134],{"class":1041,"line":1133},5,[1039,1135,1136],{"class":1095},"    \n",[1039,1138,1140],{"class":1041,"line":1139},6,[1039,1141,1142],{"class":1045},"    \u002F\u002F 准备阶段：直接赋值 123（final 常量）\n",[1039,1144,1146,1148,1150,1153,1155,1158,1160,1162],{"class":1041,"line":1145},7,[1039,1147,1111],{"class":1084},[1039,1149,1114],{"class":1084},[1039,1151,1152],{"class":1084}," final",[1039,1154,1117],{"class":1084},[1039,1156,1157],{"class":1095}," CONST ",[1039,1159,1123],{"class":1084},[1039,1161,1127],{"class":1126},[1039,1163,1130],{"class":1095},[1039,1165,1167],{"class":1041,"line":1166},8,[1039,1168,1169],{"class":1095},"}\n",[922,1171,1172],{"id":1172},"各类型初始值",[977,1174,1175,1185],{},[980,1176,1177],{},[983,1178,1179,1182],{},[986,1180,1181],{},"数据类型",[986,1183,1184],{},"初始值",[993,1186,1187,1195,1203,1211,1219,1227,1235,1243,1251],{},[983,1188,1189,1192],{},[998,1190,1191],{},"int",[998,1193,1194],{},"0",[983,1196,1197,1200],{},[998,1198,1199],{},"long",[998,1201,1202],{},"0L",[983,1204,1205,1208],{},[998,1206,1207],{},"short",[998,1209,1210],{},"(short) 0",[983,1212,1213,1216],{},[998,1214,1215],{},"char",[998,1217,1218],{},"'\\u0000'",[983,1220,1221,1224],{},[998,1222,1223],{},"byte",[998,1225,1226],{},"(byte) 0",[983,1228,1229,1232],{},[998,1230,1231],{},"boolean",[998,1233,1234],{},"false",[983,1236,1237,1240],{},[998,1238,1239],{},"float",[998,1241,1242],{},"0.0f",[983,1244,1245,1248],{},[998,1246,1247],{},"double",[998,1249,1250],{},"0.0d",[983,1252,1253,1256],{},[998,1254,1255],{},"reference",[998,1257,1258],{},"null",[901,1260,1262],{"id":1261},"解析resolution","解析（Resolution）",[892,1264,1265],{},"将常量池中的符号引用替换为直接引用。",[922,1267,1269],{"id":1268},"符号引用-vs-直接引用","符号引用 vs 直接引用",[977,1271,1272,1281],{},[980,1273,1274],{},[983,1275,1276,1279],{},[986,1277,1278],{},"类型",[986,1280,991],{},[993,1282,1283,1291],{},[983,1284,1285,1288],{},[998,1286,1287],{},"符号引用",[998,1289,1290],{},"用一组符号描述目标，如类名、方法名",[983,1292,1293,1296],{},[998,1294,1295],{},"直接引用",[998,1297,1298],{},"直接指向目标的指针、偏移量或句柄",[922,1300,1301],{"id":1301},"解析类型",[946,1303,1304,1307,1310,1313],{},[929,1305,1306],{},"类或接口的解析",[929,1308,1309],{},"字段解析",[929,1311,1312],{},"类方法解析",[929,1314,1315],{},"接口方法解析",[901,1317,1319],{"id":1318},"初始化initialization","初始化（Initialization）",[892,1321,1322,1323,1326],{},"执行类构造器 ",[896,1324,1325],{},"\u003Cclinit>()"," 方法的过程。",[922,1328,1330,1332],{"id":1329},"clinit-方法",[896,1331,1325],{}," 方法",[905,1334,1336],{"className":1033,"code":1335,"language":1035,"meta":11,"style":11},"public class InitDemo {\n    static int a = 1;                 \u002F\u002F ①\n    static int b;\n    \n    static {                          \u002F\u002F ②\n        b = 2;\n        System.out.println(\"静态代码块执行\");\n    }\n    \n    \u002F\u002F \u003Cclinit>() 方法由编译器收集 ① 和 ② 合并生成\n}\n",[896,1337,1338,1349,1370,1379,1383,1393,1405,1423,1428,1433,1439],{"__ignoreMap":11},[1039,1339,1340,1342,1344,1347],{"class":1041,"line":1042},[1039,1341,1085],{"class":1084},[1039,1343,1088],{"class":1084},[1039,1345,1346],{"class":1091}," InitDemo",[1039,1348,1096],{"class":1095},[1039,1350,1351,1354,1356,1359,1361,1364,1367],{"class":1041,"line":1049},[1039,1352,1353],{"class":1084},"    static",[1039,1355,1117],{"class":1084},[1039,1357,1358],{"class":1095}," a ",[1039,1360,1123],{"class":1084},[1039,1362,1363],{"class":1126}," 1",[1039,1365,1366],{"class":1095},";                 ",[1039,1368,1369],{"class":1045},"\u002F\u002F ①\n",[1039,1371,1372,1374,1376],{"class":1041,"line":1055},[1039,1373,1353],{"class":1084},[1039,1375,1117],{"class":1084},[1039,1377,1378],{"class":1095}," b;\n",[1039,1380,1381],{"class":1041,"line":1061},[1039,1382,1136],{"class":1095},[1039,1384,1385,1387,1390],{"class":1041,"line":1133},[1039,1386,1353],{"class":1084},[1039,1388,1389],{"class":1095}," {                          ",[1039,1391,1392],{"class":1045},"\u002F\u002F ②\n",[1039,1394,1395,1398,1400,1403],{"class":1041,"line":1139},[1039,1396,1397],{"class":1095},"        b ",[1039,1399,1123],{"class":1084},[1039,1401,1402],{"class":1126}," 2",[1039,1404,1130],{"class":1095},[1039,1406,1407,1410,1413,1416,1420],{"class":1041,"line":1145},[1039,1408,1409],{"class":1095},"        System.out.",[1039,1411,1412],{"class":1091},"println",[1039,1414,1415],{"class":1095},"(",[1039,1417,1419],{"class":1418},"sIIMD","\"静态代码块执行\"",[1039,1421,1422],{"class":1095},");\n",[1039,1424,1425],{"class":1041,"line":1166},[1039,1426,1427],{"class":1095},"    }\n",[1039,1429,1431],{"class":1041,"line":1430},9,[1039,1432,1136],{"class":1095},[1039,1434,1436],{"class":1041,"line":1435},10,[1039,1437,1438],{"class":1045},"    \u002F\u002F \u003Cclinit>() 方法由编译器收集 ① 和 ② 合并生成\n",[1039,1440,1442],{"class":1041,"line":1441},11,[1039,1443,1169],{"class":1095},[922,1445,1446],{"id":1446},"初始化时机",[892,1448,1449],{},"以下情况会触发类的初始化：",[926,1451,1452,1458,1464,1469,1479,1484],{},[929,1453,1454],{},[1455,1456,1457],"strong",{},"new 实例化对象",[929,1459,1460,1463],{},[1455,1461,1462],{},"访问类的静态变量","（非 final 常量）",[929,1465,1466],{},[1455,1467,1468],{},"调用类的静态方法",[929,1470,1471,1474,1475,1478],{},[1455,1472,1473],{},"反射调用","（如 ",[896,1476,1477],{},"Class.forName()","）",[929,1480,1481],{},[1455,1482,1483],{},"初始化子类时，父类先初始化",[929,1485,1486],{},[1455,1487,1488],{},"main 方法所在的类",[922,1490,1491],{"id":1491},"不会触发初始化的情况",[905,1493,1495],{"className":1033,"code":1494,"language":1035,"meta":11,"style":11},"public class Parent {\n    static {\n        System.out.println(\"Parent 初始化\");\n    }\n    public static int value = 123;\n}\n\npublic class Child extends Parent {\n    static {\n        System.out.println(\"Child 初始化\");\n    }\n}\n\npublic class Test {\n    public static void main(String[] args) {\n        \u002F\u002F 1. 通过子类引用父类的静态变量，不会初始化子类\n        System.out.println(Child.value);\n        \u002F\u002F 输出：Parent 初始化\n        \u002F\u002F       123\n        \n        \u002F\u002F 2. 通过数组定义，不会触发初始化\n        Parent[] arr = new Parent[10];\n        \n        \u002F\u002F 3. 引用常量不会触发初始化（常量在编译期放入常量池）\n        System.out.println(Parent.CONST);\n    }\n}\n",[896,1496,1497,1508,1514,1527,1531,1547,1551,1557,1573,1579,1592,1596,1601,1606,1618,1646,1652,1662,1668,1674,1680,1686,1711,1716,1722,1732,1737],{"__ignoreMap":11},[1039,1498,1499,1501,1503,1506],{"class":1041,"line":1042},[1039,1500,1085],{"class":1084},[1039,1502,1088],{"class":1084},[1039,1504,1505],{"class":1091}," Parent",[1039,1507,1096],{"class":1095},[1039,1509,1510,1512],{"class":1041,"line":1049},[1039,1511,1353],{"class":1084},[1039,1513,1096],{"class":1095},[1039,1515,1516,1518,1520,1522,1525],{"class":1041,"line":1055},[1039,1517,1409],{"class":1095},[1039,1519,1412],{"class":1091},[1039,1521,1415],{"class":1095},[1039,1523,1524],{"class":1418},"\"Parent 初始化\"",[1039,1526,1422],{"class":1095},[1039,1528,1529],{"class":1041,"line":1061},[1039,1530,1427],{"class":1095},[1039,1532,1533,1535,1537,1539,1541,1543,1545],{"class":1041,"line":1133},[1039,1534,1111],{"class":1084},[1039,1536,1114],{"class":1084},[1039,1538,1117],{"class":1084},[1039,1540,1120],{"class":1095},[1039,1542,1123],{"class":1084},[1039,1544,1127],{"class":1126},[1039,1546,1130],{"class":1095},[1039,1548,1549],{"class":1041,"line":1139},[1039,1550,1169],{"class":1095},[1039,1552,1553],{"class":1041,"line":1145},[1039,1554,1556],{"emptyLinePlaceholder":1555},true,"\n",[1039,1558,1559,1561,1563,1566,1569,1571],{"class":1041,"line":1166},[1039,1560,1085],{"class":1084},[1039,1562,1088],{"class":1084},[1039,1564,1565],{"class":1091}," Child",[1039,1567,1568],{"class":1084}," extends",[1039,1570,1505],{"class":1091},[1039,1572,1096],{"class":1095},[1039,1574,1575,1577],{"class":1041,"line":1430},[1039,1576,1353],{"class":1084},[1039,1578,1096],{"class":1095},[1039,1580,1581,1583,1585,1587,1590],{"class":1041,"line":1435},[1039,1582,1409],{"class":1095},[1039,1584,1412],{"class":1091},[1039,1586,1415],{"class":1095},[1039,1588,1589],{"class":1418},"\"Child 初始化\"",[1039,1591,1422],{"class":1095},[1039,1593,1594],{"class":1041,"line":1441},[1039,1595,1427],{"class":1095},[1039,1597,1599],{"class":1041,"line":1598},12,[1039,1600,1169],{"class":1095},[1039,1602,1604],{"class":1041,"line":1603},13,[1039,1605,1556],{"emptyLinePlaceholder":1555},[1039,1607,1609,1611,1613,1616],{"class":1041,"line":1608},14,[1039,1610,1085],{"class":1084},[1039,1612,1088],{"class":1084},[1039,1614,1615],{"class":1091}," Test",[1039,1617,1096],{"class":1095},[1039,1619,1621,1623,1625,1628,1631,1633,1636,1639,1643],{"class":1041,"line":1620},15,[1039,1622,1111],{"class":1084},[1039,1624,1114],{"class":1084},[1039,1626,1627],{"class":1084}," void",[1039,1629,1630],{"class":1091}," main",[1039,1632,1415],{"class":1095},[1039,1634,1635],{"class":1084},"String",[1039,1637,1638],{"class":1095},"[] ",[1039,1640,1642],{"class":1641},"sP4rz","args",[1039,1644,1645],{"class":1095},") {\n",[1039,1647,1649],{"class":1041,"line":1648},16,[1039,1650,1651],{"class":1045},"        \u002F\u002F 1. 通过子类引用父类的静态变量，不会初始化子类\n",[1039,1653,1655,1657,1659],{"class":1041,"line":1654},17,[1039,1656,1409],{"class":1095},[1039,1658,1412],{"class":1091},[1039,1660,1661],{"class":1095},"(Child.value);\n",[1039,1663,1665],{"class":1041,"line":1664},18,[1039,1666,1667],{"class":1045},"        \u002F\u002F 输出：Parent 初始化\n",[1039,1669,1671],{"class":1041,"line":1670},19,[1039,1672,1673],{"class":1045},"        \u002F\u002F       123\n",[1039,1675,1677],{"class":1041,"line":1676},20,[1039,1678,1679],{"class":1095},"        \n",[1039,1681,1683],{"class":1041,"line":1682},21,[1039,1684,1685],{"class":1045},"        \u002F\u002F 2. 通过数组定义，不会触发初始化\n",[1039,1687,1689,1692,1695,1697,1700,1702,1705,1708],{"class":1041,"line":1688},22,[1039,1690,1691],{"class":1084},"        Parent",[1039,1693,1694],{"class":1095},"[] arr ",[1039,1696,1123],{"class":1084},[1039,1698,1699],{"class":1084}," new",[1039,1701,1505],{"class":1084},[1039,1703,1704],{"class":1095},"[",[1039,1706,1707],{"class":1126},"10",[1039,1709,1710],{"class":1095},"];\n",[1039,1712,1714],{"class":1041,"line":1713},23,[1039,1715,1679],{"class":1095},[1039,1717,1719],{"class":1041,"line":1718},24,[1039,1720,1721],{"class":1045},"        \u002F\u002F 3. 引用常量不会触发初始化（常量在编译期放入常量池）\n",[1039,1723,1725,1727,1729],{"class":1041,"line":1724},25,[1039,1726,1409],{"class":1095},[1039,1728,1412],{"class":1091},[1039,1730,1731],{"class":1095},"(Parent.CONST);\n",[1039,1733,1735],{"class":1041,"line":1734},26,[1039,1736,1427],{"class":1095},[1039,1738,1740],{"class":1041,"line":1739},27,[1039,1741,1169],{"class":1095},[901,1743,1744],{"id":1744},"类加载器",[922,1746,1747],{"id":1747},"类加载器类型",[905,1749,1752],{"className":1750,"code":1751,"language":910,"meta":911},[908],"flowchart TD\n    Boot[\"Bootstrap ClassLoader\u003Cbr\u002F>启动类加载器 #40;C++ 实现#41;\u003Cbr\u002F>加载 JAVA_HOME\u002Flib 下的类\"]\n    Ext[\"Extension ClassLoader\u003Cbr\u002F>扩展类加载器\u003Cbr\u002F>加载 JAVA_HOME\u002Flib\u002Fext 下的类\"]\n    App[\"Application ClassLoader\u003Cbr\u002F>应用程序类加载器\u003Cbr\u002F>加载用户类路径 #40;classpath#41; 下的类\"]\n    Custom[\"Custom ClassLoader\u003Cbr\u002F>自定义类加载器\"]\n    Boot --> Ext --> App --> Custom\n",[896,1753,1751],{"__ignoreMap":11},[922,1755,1756],{"id":1756},"获取类加载器",[905,1758,1760],{"className":1033,"code":1759,"language":1035,"meta":11,"style":11},"public class ClassLoaderDemo {\n    public static void main(String[] args) {\n        \u002F\u002F 应用程序类加载器\n        ClassLoader appLoader = ClassLoaderDemo.class.getClassLoader();\n        System.out.println(appLoader);\n        \u002F\u002F sun.misc.Launcher$AppClassLoader\n        \n        \u002F\u002F 扩展类加载器\n        ClassLoader extLoader = appLoader.getParent();\n        System.out.println(extLoader);\n        \u002F\u002F sun.misc.Launcher$ExtClassLoader\n        \n        \u002F\u002F 启动类加载器（C++ 实现，返回 null）\n        ClassLoader bootstrapLoader = extLoader.getParent();\n        System.out.println(bootstrapLoader);\n        \u002F\u002F null\n        \n        \u002F\u002F String 类由启动类加载器加载\n        System.out.println(String.class.getClassLoader());\n        \u002F\u002F null\n    }\n}\n",[896,1761,1762,1773,1793,1798,1814,1823,1828,1832,1837,1852,1861,1866,1870,1875,1889,1898,1903,1907,1912,1926,1930,1934],{"__ignoreMap":11},[1039,1763,1764,1766,1768,1771],{"class":1041,"line":1042},[1039,1765,1085],{"class":1084},[1039,1767,1088],{"class":1084},[1039,1769,1770],{"class":1091}," ClassLoaderDemo",[1039,1772,1096],{"class":1095},[1039,1774,1775,1777,1779,1781,1783,1785,1787,1789,1791],{"class":1041,"line":1049},[1039,1776,1111],{"class":1084},[1039,1778,1114],{"class":1084},[1039,1780,1627],{"class":1084},[1039,1782,1630],{"class":1091},[1039,1784,1415],{"class":1095},[1039,1786,1635],{"class":1084},[1039,1788,1638],{"class":1095},[1039,1790,1642],{"class":1641},[1039,1792,1645],{"class":1095},[1039,1794,1795],{"class":1041,"line":1055},[1039,1796,1797],{"class":1045},"        \u002F\u002F 应用程序类加载器\n",[1039,1799,1800,1803,1805,1808,1811],{"class":1041,"line":1061},[1039,1801,1802],{"class":1095},"        ClassLoader appLoader ",[1039,1804,1123],{"class":1084},[1039,1806,1807],{"class":1095}," ClassLoaderDemo.class.",[1039,1809,1810],{"class":1091},"getClassLoader",[1039,1812,1813],{"class":1095},"();\n",[1039,1815,1816,1818,1820],{"class":1041,"line":1133},[1039,1817,1409],{"class":1095},[1039,1819,1412],{"class":1091},[1039,1821,1822],{"class":1095},"(appLoader);\n",[1039,1824,1825],{"class":1041,"line":1139},[1039,1826,1827],{"class":1045},"        \u002F\u002F sun.misc.Launcher$AppClassLoader\n",[1039,1829,1830],{"class":1041,"line":1145},[1039,1831,1679],{"class":1095},[1039,1833,1834],{"class":1041,"line":1166},[1039,1835,1836],{"class":1045},"        \u002F\u002F 扩展类加载器\n",[1039,1838,1839,1842,1844,1847,1850],{"class":1041,"line":1430},[1039,1840,1841],{"class":1095},"        ClassLoader extLoader ",[1039,1843,1123],{"class":1084},[1039,1845,1846],{"class":1095}," appLoader.",[1039,1848,1849],{"class":1091},"getParent",[1039,1851,1813],{"class":1095},[1039,1853,1854,1856,1858],{"class":1041,"line":1435},[1039,1855,1409],{"class":1095},[1039,1857,1412],{"class":1091},[1039,1859,1860],{"class":1095},"(extLoader);\n",[1039,1862,1863],{"class":1041,"line":1441},[1039,1864,1865],{"class":1045},"        \u002F\u002F sun.misc.Launcher$ExtClassLoader\n",[1039,1867,1868],{"class":1041,"line":1598},[1039,1869,1679],{"class":1095},[1039,1871,1872],{"class":1041,"line":1603},[1039,1873,1874],{"class":1045},"        \u002F\u002F 启动类加载器（C++ 实现，返回 null）\n",[1039,1876,1877,1880,1882,1885,1887],{"class":1041,"line":1608},[1039,1878,1879],{"class":1095},"        ClassLoader bootstrapLoader ",[1039,1881,1123],{"class":1084},[1039,1883,1884],{"class":1095}," extLoader.",[1039,1886,1849],{"class":1091},[1039,1888,1813],{"class":1095},[1039,1890,1891,1893,1895],{"class":1041,"line":1620},[1039,1892,1409],{"class":1095},[1039,1894,1412],{"class":1091},[1039,1896,1897],{"class":1095},"(bootstrapLoader);\n",[1039,1899,1900],{"class":1041,"line":1648},[1039,1901,1902],{"class":1045},"        \u002F\u002F null\n",[1039,1904,1905],{"class":1041,"line":1654},[1039,1906,1679],{"class":1095},[1039,1908,1909],{"class":1041,"line":1664},[1039,1910,1911],{"class":1045},"        \u002F\u002F String 类由启动类加载器加载\n",[1039,1913,1914,1916,1918,1921,1923],{"class":1041,"line":1670},[1039,1915,1409],{"class":1095},[1039,1917,1412],{"class":1091},[1039,1919,1920],{"class":1095},"(String.class.",[1039,1922,1810],{"class":1091},[1039,1924,1925],{"class":1095},"());\n",[1039,1927,1928],{"class":1041,"line":1676},[1039,1929,1902],{"class":1045},[1039,1931,1932],{"class":1041,"line":1682},[1039,1933,1427],{"class":1095},[1039,1935,1936],{"class":1041,"line":1688},[1039,1937,1169],{"class":1095},[901,1939,1940],{"id":1940},"双亲委派模型",[922,1942,1943],{"id":1943},"工作原理",[905,1945,1948],{"className":1946,"code":1947,"language":910},[908],"加载请求\n    ↓\nApplication ClassLoader → 委托给 Extension ClassLoader\n                                    ↓\n              Extension ClassLoader → 委托给 Bootstrap ClassLoader\n                                              ↓\n                              Bootstrap 尝试加载\n                              （成功则返回，失败则下传）\n                                              ↓\n                              Extension 尝试加载\n                              （成功则返回，失败则下传）\n                                              ↓\n                              Application 尝试加载\n",[896,1949,1947],{"__ignoreMap":11},[922,1951,1952],{"id":1952},"源码分析",[905,1954,1956],{"className":1033,"code":1955,"language":1035,"meta":11,"style":11},"protected Class\u003C?> loadClass(String name, boolean resolve) throws ClassNotFoundException {\n    synchronized (getClassLoadingLock(name)) {\n        \u002F\u002F 1. 检查类是否已加载\n        Class\u003C?> c = findLoadedClass(name);\n        if (c == null) {\n            try {\n                \u002F\u002F 2. 委托给父加载器\n                if (parent != null) {\n                    c = parent.loadClass(name, false);\n                } else {\n                    \u002F\u002F 3. 父加载器为空，使用启动类加载器\n                    c = findBootstrapClassOrNull(name);\n                }\n            } catch (ClassNotFoundException e) {\n                \u002F\u002F 父加载器无法加载\n            }\n\n            if (c == null) {\n                \u002F\u002F 4. 父加载器无法加载，自己尝试加载\n                c = findClass(name);\n            }\n        }\n        if (resolve) {\n            resolveClass(c);\n        }\n        return c;\n    }\n}\n",[896,1957,1958,1980,1994,1999,2018,2034,2041,2046,2061,2081,2091,2096,2107,2112,2128,2133,2138,2142,2155,2160,2172,2176,2181,2188,2196,2200,2208,2212],{"__ignoreMap":11},[1039,1959,1960,1963,1966,1969,1972,1975,1977],{"class":1041,"line":1042},[1039,1961,1962],{"class":1084},"protected",[1039,1964,1965],{"class":1095}," Class",[1039,1967,1968],{"class":1084},"\u003C?>",[1039,1970,1971],{"class":1091}," loadClass",[1039,1973,1974],{"class":1095},"(String name, ",[1039,1976,1231],{"class":1084},[1039,1978,1979],{"class":1095}," resolve) throws ClassNotFoundException {\n",[1039,1981,1982,1985,1988,1991],{"class":1041,"line":1049},[1039,1983,1984],{"class":1084},"    synchronized",[1039,1986,1987],{"class":1095}," (",[1039,1989,1990],{"class":1091},"getClassLoadingLock",[1039,1992,1993],{"class":1095},"(name)) {\n",[1039,1995,1996],{"class":1041,"line":1055},[1039,1997,1998],{"class":1045},"        \u002F\u002F 1. 检查类是否已加载\n",[1039,2000,2001,2004,2007,2010,2012,2015],{"class":1041,"line":1061},[1039,2002,2003],{"class":1095},"        Class\u003C",[1039,2005,2006],{"class":1084},"?",[1039,2008,2009],{"class":1095},"> c ",[1039,2011,1123],{"class":1084},[1039,2013,2014],{"class":1091}," findLoadedClass",[1039,2016,2017],{"class":1095},"(name);\n",[1039,2019,2020,2023,2026,2029,2032],{"class":1041,"line":1133},[1039,2021,2022],{"class":1084},"        if",[1039,2024,2025],{"class":1095}," (c ",[1039,2027,2028],{"class":1084},"==",[1039,2030,2031],{"class":1126}," null",[1039,2033,1645],{"class":1095},[1039,2035,2036,2039],{"class":1041,"line":1139},[1039,2037,2038],{"class":1084},"            try",[1039,2040,1096],{"class":1095},[1039,2042,2043],{"class":1041,"line":1145},[1039,2044,2045],{"class":1045},"                \u002F\u002F 2. 委托给父加载器\n",[1039,2047,2048,2051,2054,2057,2059],{"class":1041,"line":1166},[1039,2049,2050],{"class":1084},"                if",[1039,2052,2053],{"class":1095}," (parent ",[1039,2055,2056],{"class":1084},"!=",[1039,2058,2031],{"class":1126},[1039,2060,1645],{"class":1095},[1039,2062,2063,2066,2068,2071,2074,2077,2079],{"class":1041,"line":1430},[1039,2064,2065],{"class":1095},"                    c ",[1039,2067,1123],{"class":1084},[1039,2069,2070],{"class":1095}," parent.",[1039,2072,2073],{"class":1091},"loadClass",[1039,2075,2076],{"class":1095},"(name, ",[1039,2078,1234],{"class":1126},[1039,2080,1422],{"class":1095},[1039,2082,2083,2086,2089],{"class":1041,"line":1435},[1039,2084,2085],{"class":1095},"                } ",[1039,2087,2088],{"class":1084},"else",[1039,2090,1096],{"class":1095},[1039,2092,2093],{"class":1041,"line":1441},[1039,2094,2095],{"class":1045},"                    \u002F\u002F 3. 父加载器为空，使用启动类加载器\n",[1039,2097,2098,2100,2102,2105],{"class":1041,"line":1598},[1039,2099,2065],{"class":1095},[1039,2101,1123],{"class":1084},[1039,2103,2104],{"class":1091}," findBootstrapClassOrNull",[1039,2106,2017],{"class":1095},[1039,2108,2109],{"class":1041,"line":1603},[1039,2110,2111],{"class":1095},"                }\n",[1039,2113,2114,2117,2120,2123,2126],{"class":1041,"line":1608},[1039,2115,2116],{"class":1095},"            } ",[1039,2118,2119],{"class":1084},"catch",[1039,2121,2122],{"class":1095}," (ClassNotFoundException ",[1039,2124,2125],{"class":1641},"e",[1039,2127,1645],{"class":1095},[1039,2129,2130],{"class":1041,"line":1620},[1039,2131,2132],{"class":1045},"                \u002F\u002F 父加载器无法加载\n",[1039,2134,2135],{"class":1041,"line":1648},[1039,2136,2137],{"class":1095},"            }\n",[1039,2139,2140],{"class":1041,"line":1654},[1039,2141,1556],{"emptyLinePlaceholder":1555},[1039,2143,2144,2147,2149,2151,2153],{"class":1041,"line":1664},[1039,2145,2146],{"class":1084},"            if",[1039,2148,2025],{"class":1095},[1039,2150,2028],{"class":1084},[1039,2152,2031],{"class":1126},[1039,2154,1645],{"class":1095},[1039,2156,2157],{"class":1041,"line":1670},[1039,2158,2159],{"class":1045},"                \u002F\u002F 4. 父加载器无法加载，自己尝试加载\n",[1039,2161,2162,2165,2167,2170],{"class":1041,"line":1676},[1039,2163,2164],{"class":1095},"                c ",[1039,2166,1123],{"class":1084},[1039,2168,2169],{"class":1091}," findClass",[1039,2171,2017],{"class":1095},[1039,2173,2174],{"class":1041,"line":1682},[1039,2175,2137],{"class":1095},[1039,2177,2178],{"class":1041,"line":1688},[1039,2179,2180],{"class":1095},"        }\n",[1039,2182,2183,2185],{"class":1041,"line":1713},[1039,2184,2022],{"class":1084},[1039,2186,2187],{"class":1095}," (resolve) {\n",[1039,2189,2190,2193],{"class":1041,"line":1718},[1039,2191,2192],{"class":1091},"            resolveClass",[1039,2194,2195],{"class":1095},"(c);\n",[1039,2197,2198],{"class":1041,"line":1724},[1039,2199,2180],{"class":1095},[1039,2201,2202,2205],{"class":1041,"line":1734},[1039,2203,2204],{"class":1084},"        return",[1039,2206,2207],{"class":1095}," c;\n",[1039,2209,2210],{"class":1041,"line":1739},[1039,2211,1427],{"class":1095},[1039,2213,2215],{"class":1041,"line":2214},28,[1039,2216,1169],{"class":1095},[922,2218,2219],{"id":2219},"双亲委派的优点",[926,2221,2222,2227],{},[929,2223,2224],{},[1455,2225,2226],{},"避免类的重复加载",[929,2228,2229,2232],{},[1455,2230,2231],{},"保证核心类的安全","：防止用户自定义类覆盖核心类",[905,2234,2236],{"className":1033,"code":2235,"language":1035,"meta":11,"style":11},"\u002F\u002F 即使自定义 java.lang.String，也不会加载\n\u002F\u002F 因为会委托给 Bootstrap ClassLoader，它会加载 rt.jar 中的 String\npackage java.lang;\npublic class String {\n    \u002F\u002F 这个类不会被加载\n}\n",[896,2237,2238,2243,2248,2256,2267,2272],{"__ignoreMap":11},[1039,2239,2240],{"class":1041,"line":1042},[1039,2241,2242],{"class":1045},"\u002F\u002F 即使自定义 java.lang.String，也不会加载\n",[1039,2244,2245],{"class":1041,"line":1049},[1039,2246,2247],{"class":1045},"\u002F\u002F 因为会委托给 Bootstrap ClassLoader，它会加载 rt.jar 中的 String\n",[1039,2249,2250,2253],{"class":1041,"line":1055},[1039,2251,2252],{"class":1084},"package",[1039,2254,2255],{"class":1095}," java.lang;\n",[1039,2257,2258,2260,2262,2265],{"class":1041,"line":1061},[1039,2259,1085],{"class":1084},[1039,2261,1088],{"class":1084},[1039,2263,2264],{"class":1091}," String",[1039,2266,1096],{"class":1095},[1039,2268,2269],{"class":1041,"line":1133},[1039,2270,2271],{"class":1045},"    \u002F\u002F 这个类不会被加载\n",[1039,2273,2274],{"class":1041,"line":1139},[1039,2275,1169],{"class":1095},[901,2277,2278],{"id":2278},"自定义类加载器",[922,2280,2281],{"id":2281},"实现方式",[892,2283,2284,2285,2288,2289,2292],{},"继承 ",[896,2286,2287],{},"ClassLoader"," 并重写 ",[896,2290,2291],{},"findClass"," 方法。",[905,2294,2296],{"className":1033,"code":2295,"language":1035,"meta":11,"style":11},"public class MyClassLoader extends ClassLoader {\n    private String classPath;\n    \n    public MyClassLoader(String classPath) {\n        this.classPath = classPath;\n    }\n    \n    @Override\n    protected Class\u003C?> findClass(String name) throws ClassNotFoundException {\n        try {\n            \u002F\u002F 读取 class 文件\n            byte[] bytes = loadClassBytes(name);\n            \u002F\u002F 定义类\n            return defineClass(name, bytes, 0, bytes.length);\n        } catch (IOException e) {\n            throw new ClassNotFoundException(name, e);\n        }\n    }\n    \n    private byte[] loadClassBytes(String name) throws IOException {\n        String path = classPath + \"\u002F\" + name.replace('.', '\u002F') + \".class\";\n        try (InputStream is = new FileInputStream(path);\n             ByteArrayOutputStream baos = new ByteArrayOutputStream()) {\n            byte[] buffer = new byte[1024];\n            int len;\n            while ((len = is.read(buffer)) != -1) {\n                baos.write(buffer, 0, len);\n            }\n            return baos.toByteArray();\n        }\n    }\n}\n",[896,2297,2298,2314,2322,2326,2340,2353,2357,2361,2369,2398,2405,2410,2425,2430,2446,2460,2473,2477,2481,2485,2508,2553,2570,2585,2605,2613,2642,2658,2662,2675,2680,2685],{"__ignoreMap":11},[1039,2299,2300,2302,2304,2307,2309,2312],{"class":1041,"line":1042},[1039,2301,1085],{"class":1084},[1039,2303,1088],{"class":1084},[1039,2305,2306],{"class":1091}," MyClassLoader",[1039,2308,1568],{"class":1084},[1039,2310,2311],{"class":1091}," ClassLoader",[1039,2313,1096],{"class":1095},[1039,2315,2316,2319],{"class":1041,"line":1049},[1039,2317,2318],{"class":1084},"    private",[1039,2320,2321],{"class":1095}," String classPath;\n",[1039,2323,2324],{"class":1041,"line":1055},[1039,2325,1136],{"class":1095},[1039,2327,2328,2330,2332,2335,2338],{"class":1041,"line":1061},[1039,2329,1111],{"class":1084},[1039,2331,2306],{"class":1091},[1039,2333,2334],{"class":1095},"(String ",[1039,2336,2337],{"class":1641},"classPath",[1039,2339,1645],{"class":1095},[1039,2341,2342,2345,2348,2350],{"class":1041,"line":1133},[1039,2343,2344],{"class":1126},"        this",[1039,2346,2347],{"class":1095},".classPath ",[1039,2349,1123],{"class":1084},[1039,2351,2352],{"class":1095}," classPath;\n",[1039,2354,2355],{"class":1041,"line":1139},[1039,2356,1427],{"class":1095},[1039,2358,2359],{"class":1041,"line":1145},[1039,2360,1136],{"class":1095},[1039,2362,2363,2366],{"class":1041,"line":1166},[1039,2364,2365],{"class":1095},"    @",[1039,2367,2368],{"class":1084},"Override\n",[1039,2370,2371,2374,2377,2379,2382,2384,2386,2389,2392,2395],{"class":1041,"line":1430},[1039,2372,2373],{"class":1084},"    protected",[1039,2375,2376],{"class":1095}," Class\u003C",[1039,2378,2006],{"class":1084},[1039,2380,2381],{"class":1095},"> ",[1039,2383,2291],{"class":1091},[1039,2385,2334],{"class":1095},[1039,2387,2388],{"class":1641},"name",[1039,2390,2391],{"class":1095},") ",[1039,2393,2394],{"class":1084},"throws",[1039,2396,2397],{"class":1095}," ClassNotFoundException {\n",[1039,2399,2400,2403],{"class":1041,"line":1435},[1039,2401,2402],{"class":1084},"        try",[1039,2404,1096],{"class":1095},[1039,2406,2407],{"class":1041,"line":1441},[1039,2408,2409],{"class":1045},"            \u002F\u002F 读取 class 文件\n",[1039,2411,2412,2415,2418,2420,2423],{"class":1041,"line":1598},[1039,2413,2414],{"class":1084},"            byte",[1039,2416,2417],{"class":1095},"[] bytes ",[1039,2419,1123],{"class":1084},[1039,2421,2422],{"class":1091}," loadClassBytes",[1039,2424,2017],{"class":1095},[1039,2426,2427],{"class":1041,"line":1603},[1039,2428,2429],{"class":1045},"            \u002F\u002F 定义类\n",[1039,2431,2432,2435,2438,2441,2443],{"class":1041,"line":1608},[1039,2433,2434],{"class":1084},"            return",[1039,2436,2437],{"class":1091}," defineClass",[1039,2439,2440],{"class":1095},"(name, bytes, ",[1039,2442,1194],{"class":1126},[1039,2444,2445],{"class":1095},", bytes.length);\n",[1039,2447,2448,2451,2453,2456,2458],{"class":1041,"line":1620},[1039,2449,2450],{"class":1095},"        } ",[1039,2452,2119],{"class":1084},[1039,2454,2455],{"class":1095}," (IOException ",[1039,2457,2125],{"class":1641},[1039,2459,1645],{"class":1095},[1039,2461,2462,2465,2467,2470],{"class":1041,"line":1648},[1039,2463,2464],{"class":1084},"            throw",[1039,2466,1699],{"class":1084},[1039,2468,2469],{"class":1091}," ClassNotFoundException",[1039,2471,2472],{"class":1095},"(name, e);\n",[1039,2474,2475],{"class":1041,"line":1654},[1039,2476,2180],{"class":1095},[1039,2478,2479],{"class":1041,"line":1664},[1039,2480,1427],{"class":1095},[1039,2482,2483],{"class":1041,"line":1670},[1039,2484,1136],{"class":1095},[1039,2486,2487,2489,2492,2494,2497,2499,2501,2503,2505],{"class":1041,"line":1676},[1039,2488,2318],{"class":1084},[1039,2490,2491],{"class":1084}," byte",[1039,2493,1638],{"class":1095},[1039,2495,2496],{"class":1091},"loadClassBytes",[1039,2498,2334],{"class":1095},[1039,2500,2388],{"class":1641},[1039,2502,2391],{"class":1095},[1039,2504,2394],{"class":1084},[1039,2506,2507],{"class":1095}," IOException {\n",[1039,2509,2510,2513,2515,2518,2521,2524,2527,2530,2533,2535,2538,2541,2544,2546,2548,2551],{"class":1041,"line":1682},[1039,2511,2512],{"class":1095},"        String path ",[1039,2514,1123],{"class":1084},[1039,2516,2517],{"class":1095}," classPath ",[1039,2519,2520],{"class":1084},"+",[1039,2522,2523],{"class":1418}," \"\u002F\"",[1039,2525,2526],{"class":1084}," +",[1039,2528,2529],{"class":1095}," name.",[1039,2531,2532],{"class":1091},"replace",[1039,2534,1415],{"class":1095},[1039,2536,2537],{"class":1418},"'.'",[1039,2539,2540],{"class":1095},", ",[1039,2542,2543],{"class":1418},"'\u002F'",[1039,2545,2391],{"class":1095},[1039,2547,2520],{"class":1084},[1039,2549,2550],{"class":1418}," \".class\"",[1039,2552,1130],{"class":1095},[1039,2554,2555,2557,2560,2562,2564,2567],{"class":1041,"line":1688},[1039,2556,2402],{"class":1084},[1039,2558,2559],{"class":1095}," (InputStream is ",[1039,2561,1123],{"class":1084},[1039,2563,1699],{"class":1084},[1039,2565,2566],{"class":1091}," FileInputStream",[1039,2568,2569],{"class":1095},"(path);\n",[1039,2571,2572,2575,2577,2579,2582],{"class":1041,"line":1713},[1039,2573,2574],{"class":1095},"             ByteArrayOutputStream baos ",[1039,2576,1123],{"class":1084},[1039,2578,1699],{"class":1084},[1039,2580,2581],{"class":1091}," ByteArrayOutputStream",[1039,2583,2584],{"class":1095},"()) {\n",[1039,2586,2587,2589,2592,2594,2596,2598,2600,2603],{"class":1041,"line":1718},[1039,2588,2414],{"class":1084},[1039,2590,2591],{"class":1095},"[] buffer ",[1039,2593,1123],{"class":1084},[1039,2595,1699],{"class":1084},[1039,2597,2491],{"class":1084},[1039,2599,1704],{"class":1095},[1039,2601,2602],{"class":1126},"1024",[1039,2604,1710],{"class":1095},[1039,2606,2607,2610],{"class":1041,"line":1724},[1039,2608,2609],{"class":1084},"            int",[1039,2611,2612],{"class":1095}," len;\n",[1039,2614,2615,2618,2621,2623,2626,2629,2632,2634,2637,2640],{"class":1041,"line":1734},[1039,2616,2617],{"class":1084},"            while",[1039,2619,2620],{"class":1095}," ((len ",[1039,2622,1123],{"class":1084},[1039,2624,2625],{"class":1095}," is.",[1039,2627,2628],{"class":1091},"read",[1039,2630,2631],{"class":1095},"(buffer)) ",[1039,2633,2056],{"class":1084},[1039,2635,2636],{"class":1084}," -",[1039,2638,2639],{"class":1126},"1",[1039,2641,1645],{"class":1095},[1039,2643,2644,2647,2650,2653,2655],{"class":1041,"line":1739},[1039,2645,2646],{"class":1095},"                baos.",[1039,2648,2649],{"class":1091},"write",[1039,2651,2652],{"class":1095},"(buffer, ",[1039,2654,1194],{"class":1126},[1039,2656,2657],{"class":1095},", len);\n",[1039,2659,2660],{"class":1041,"line":2214},[1039,2661,2137],{"class":1095},[1039,2663,2665,2667,2670,2673],{"class":1041,"line":2664},29,[1039,2666,2434],{"class":1084},[1039,2668,2669],{"class":1095}," baos.",[1039,2671,2672],{"class":1091},"toByteArray",[1039,2674,1813],{"class":1095},[1039,2676,2678],{"class":1041,"line":2677},30,[1039,2679,2180],{"class":1095},[1039,2681,2683],{"class":1041,"line":2682},31,[1039,2684,1427],{"class":1095},[1039,2686,2688],{"class":1041,"line":2687},32,[1039,2689,1169],{"class":1095},[922,2691,2692],{"id":2692},"使用示例",[905,2694,2696],{"className":1033,"code":2695,"language":1035,"meta":11,"style":11},"public class CustomLoaderTest {\n    public static void main(String[] args) throws Exception {\n        MyClassLoader loader = new MyClassLoader(\"\u002Fpath\u002Fto\u002Fclasses\");\n        Class\u003C?> clazz = loader.loadClass(\"com.example.MyClass\");\n        Object instance = clazz.newInstance();\n        System.out.println(instance.getClass().getClassLoader());\n    }\n}\n",[896,2697,2698,2709,2734,2752,2775,2790,2809,2813],{"__ignoreMap":11},[1039,2699,2700,2702,2704,2707],{"class":1041,"line":1042},[1039,2701,1085],{"class":1084},[1039,2703,1088],{"class":1084},[1039,2705,2706],{"class":1091}," CustomLoaderTest",[1039,2708,1096],{"class":1095},[1039,2710,2711,2713,2715,2717,2719,2721,2723,2725,2727,2729,2731],{"class":1041,"line":1049},[1039,2712,1111],{"class":1084},[1039,2714,1114],{"class":1084},[1039,2716,1627],{"class":1084},[1039,2718,1630],{"class":1091},[1039,2720,1415],{"class":1095},[1039,2722,1635],{"class":1084},[1039,2724,1638],{"class":1095},[1039,2726,1642],{"class":1641},[1039,2728,2391],{"class":1095},[1039,2730,2394],{"class":1084},[1039,2732,2733],{"class":1095}," Exception {\n",[1039,2735,2736,2739,2741,2743,2745,2747,2750],{"class":1041,"line":1055},[1039,2737,2738],{"class":1095},"        MyClassLoader loader ",[1039,2740,1123],{"class":1084},[1039,2742,1699],{"class":1084},[1039,2744,2306],{"class":1091},[1039,2746,1415],{"class":1095},[1039,2748,2749],{"class":1418},"\"\u002Fpath\u002Fto\u002Fclasses\"",[1039,2751,1422],{"class":1095},[1039,2753,2754,2756,2758,2761,2763,2766,2768,2770,2773],{"class":1041,"line":1061},[1039,2755,2003],{"class":1095},[1039,2757,2006],{"class":1084},[1039,2759,2760],{"class":1095},"> clazz ",[1039,2762,1123],{"class":1084},[1039,2764,2765],{"class":1095}," loader.",[1039,2767,2073],{"class":1091},[1039,2769,1415],{"class":1095},[1039,2771,2772],{"class":1418},"\"com.example.MyClass\"",[1039,2774,1422],{"class":1095},[1039,2776,2777,2780,2782,2785,2788],{"class":1041,"line":1133},[1039,2778,2779],{"class":1095},"        Object instance ",[1039,2781,1123],{"class":1084},[1039,2783,2784],{"class":1095}," clazz.",[1039,2786,2787],{"class":1091},"newInstance",[1039,2789,1813],{"class":1095},[1039,2791,2792,2794,2796,2799,2802,2805,2807],{"class":1041,"line":1139},[1039,2793,1409],{"class":1095},[1039,2795,1412],{"class":1091},[1039,2797,2798],{"class":1095},"(instance.",[1039,2800,2801],{"class":1091},"getClass",[1039,2803,2804],{"class":1095},"().",[1039,2806,1810],{"class":1091},[1039,2808,1925],{"class":1095},[1039,2810,2811],{"class":1041,"line":1145},[1039,2812,1427],{"class":1095},[1039,2814,2815],{"class":1041,"line":1166},[1039,2816,1169],{"class":1095},[901,2818,2819],{"id":2819},"打破双亲委派",[922,2821,2822],{"id":2822},"常见场景",[926,2824,2825,2831,2837],{},[929,2826,2827,2830],{},[1455,2828,2829],{},"JNDI、JDBC","：核心类需要加载 SPI 实现类",[929,2832,2833,2836],{},[1455,2834,2835],{},"热部署","：如 Tomcat 的 WebAppClassLoader",[929,2838,2839,2842],{},[1455,2840,2841],{},"模块化","：OSGi",[922,2844,2845],{"id":2845},"线程上下文类加载器",[905,2847,2849],{"className":1033,"code":2848,"language":1035,"meta":11,"style":11},"\u002F\u002F 获取线程上下文类加载器\nClassLoader contextLoader = Thread.currentThread().getContextClassLoader();\n\n\u002F\u002F 设置线程上下文类加载器\nThread.currentThread().setContextClassLoader(customLoader);\n",[896,2850,2851,2856,2876,2880,2885],{"__ignoreMap":11},[1039,2852,2853],{"class":1041,"line":1042},[1039,2854,2855],{"class":1045},"\u002F\u002F 获取线程上下文类加载器\n",[1039,2857,2858,2861,2863,2866,2869,2871,2874],{"class":1041,"line":1049},[1039,2859,2860],{"class":1095},"ClassLoader contextLoader ",[1039,2862,1123],{"class":1084},[1039,2864,2865],{"class":1095}," Thread.",[1039,2867,2868],{"class":1091},"currentThread",[1039,2870,2804],{"class":1095},[1039,2872,2873],{"class":1091},"getContextClassLoader",[1039,2875,1813],{"class":1095},[1039,2877,2878],{"class":1041,"line":1055},[1039,2879,1556],{"emptyLinePlaceholder":1555},[1039,2881,2882],{"class":1041,"line":1061},[1039,2883,2884],{"class":1045},"\u002F\u002F 设置线程上下文类加载器\n",[1039,2886,2887,2890,2892,2894,2897],{"class":1041,"line":1133},[1039,2888,2889],{"class":1095},"Thread.",[1039,2891,2868],{"class":1091},[1039,2893,2804],{"class":1095},[1039,2895,2896],{"class":1091},"setContextClassLoader",[1039,2898,2899],{"class":1095},"(customLoader);\n",[922,2901,2903],{"id":2902},"tomcat-类加载器","Tomcat 类加载器",[905,2905,2908],{"className":2906,"code":2907,"language":910},[908],"      Bootstrap\n          ↓\n       System\n          ↓\n       Common\n      ↙    ↘\n Webapp1   Webapp2\n",[896,2909,2907],{"__ignoreMap":11},[946,2911,2912,2915,2918],{},[929,2913,2914],{},"每个 Web 应用有独立的类加载器",[929,2916,2917],{},"优先加载 Web 应用自己的类",[929,2919,2920],{},"实现应用隔离",[901,2922,2923],{"id":2923},"小结",[946,2925,2926,2932,2938,2946,2952],{},[929,2927,2928,2931],{},[1455,2929,2930],{},"类加载过程","：加载 → 验证 → 准备 → 解析 → 初始化",[929,2933,2934,2937],{},[1455,2935,2936],{},"准备阶段","：静态变量赋初始值（零值），final 常量直接赋值",[929,2939,2940,2943,2944,1332],{},[1455,2941,2942],{},"初始化阶段","：执行 ",[896,2945,1325],{},[929,2947,2948,2951],{},[1455,2949,2950],{},"双亲委派","：优先委托父加载器加载，保证类加载的唯一性和安全性",[929,2953,2954,2956],{},[1455,2955,2819],{},"：热部署、SPI、模块化等场景",[2958,2959,2960],"style",{},"html pre.shiki code .sCsY4, html code.shiki .sCsY4{--shiki-light:#6A737D;--shiki-default:#6A737D;--shiki-dark:#6A737D}html .light .shiki span {color: var(--shiki-light);background: var(--shiki-light-bg);font-style: var(--shiki-light-font-style);font-weight: var(--shiki-light-font-weight);text-decoration: var(--shiki-light-text-decoration);}html.light .shiki span {color: var(--shiki-light);background: var(--shiki-light-bg);font-style: var(--shiki-light-font-style);font-weight: var(--shiki-light-font-weight);text-decoration: var(--shiki-light-text-decoration);}html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .dark .shiki span {color: var(--shiki-dark);background: var(--shiki-dark-bg);font-style: var(--shiki-dark-font-style);font-weight: var(--shiki-dark-font-weight);text-decoration: var(--shiki-dark-text-decoration);}html.dark .shiki span {color: var(--shiki-dark);background: var(--shiki-dark-bg);font-style: var(--shiki-dark-font-style);font-weight: var(--shiki-dark-font-weight);text-decoration: var(--shiki-dark-text-decoration);}html pre.shiki code .s8jYJ, html code.shiki .s8jYJ{--shiki-light:#D73A49;--shiki-default:#D73A49;--shiki-dark:#F97583}html pre.shiki code .snPdu, html code.shiki .snPdu{--shiki-light:#6F42C1;--shiki-default:#6F42C1;--shiki-dark:#B392F0}html pre.shiki code .sxrX7, html code.shiki .sxrX7{--shiki-light:#24292E;--shiki-default:#24292E;--shiki-dark:#E1E4E8}html pre.shiki code .sBjJW, html code.shiki .sBjJW{--shiki-light:#005CC5;--shiki-default:#005CC5;--shiki-dark:#79B8FF}html pre.shiki code .sIIMD, html code.shiki .sIIMD{--shiki-light:#032F62;--shiki-default:#032F62;--shiki-dark:#9ECBFF}html pre.shiki code .sP4rz, html code.shiki .sP4rz{--shiki-light:#E36209;--shiki-default:#E36209;--shiki-dark:#FFAB70}",{"title":11,"searchDepth":1049,"depth":1049,"links":2962},[2963,2964,2968,2972,2976,2980,2986,2990,2995,2999,3004],{"id":903,"depth":1049,"text":903},{"id":916,"depth":1049,"text":917,"children":2965},[2966,2967],{"id":924,"depth":1055,"text":924},{"id":944,"depth":1055,"text":944},{"id":968,"depth":1049,"text":969,"children":2969},[2970,2971],{"id":975,"depth":1055,"text":975},{"id":1030,"depth":1055,"text":1030},{"id":1067,"depth":1049,"text":1068,"children":2973},[2974,2975],{"id":1074,"depth":1055,"text":1074},{"id":1172,"depth":1055,"text":1172},{"id":1261,"depth":1049,"text":1262,"children":2977},[2978,2979],{"id":1268,"depth":1055,"text":1269},{"id":1301,"depth":1055,"text":1301},{"id":1318,"depth":1049,"text":1319,"children":2981},[2982,2984,2985],{"id":1329,"depth":1055,"text":2983},"\u003Cclinit>() 方法",{"id":1446,"depth":1055,"text":1446},{"id":1491,"depth":1055,"text":1491},{"id":1744,"depth":1049,"text":1744,"children":2987},[2988,2989],{"id":1747,"depth":1055,"text":1747},{"id":1756,"depth":1055,"text":1756},{"id":1940,"depth":1049,"text":1940,"children":2991},[2992,2993,2994],{"id":1943,"depth":1055,"text":1943},{"id":1952,"depth":1055,"text":1952},{"id":2219,"depth":1055,"text":2219},{"id":2278,"depth":1049,"text":2278,"children":2996},[2997,2998],{"id":2281,"depth":1055,"text":2281},{"id":2692,"depth":1055,"text":2692},{"id":2819,"depth":1049,"text":2819,"children":3000},[3001,3002,3003],{"id":2822,"depth":1055,"text":2822},{"id":2845,"depth":1055,"text":2845},{"id":2902,"depth":1055,"text":2903},{"id":2923,"depth":1049,"text":2923},"md",{},{"title":444,"description":445},"other\u002Fjava\u002Fjvm\u002Fclass-loading","6RD9e1WmAK5ejUs2HcJVwNzCyVTKaLlX4YEcxxObv7E",1775496428918]