[{"data":1,"prerenderedAt":3125},["ShallowReactive",2],{"search-docs":3,"doc-\u002Ftutorials\u002Ffront-end\u002Fvue3\u002Fcomponent-communication":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":816,"body":888,"description":817,"extension":3120,"meta":3121,"navigation":1246,"path":815,"seo":3122,"stem":3123,"__hash__":3124},"docs\u002Ftutorials\u002Ffront-end\u002Fvue3\u002Fcomponent-communication.md",{"type":889,"value":890,"toc":3108},"minimark",[891,894,909,915,928,939,978,983,988,995,998,1002,1005,1010,1056,1069,1072,1077,1104,1110,1604,1610,1613,1621,1624,1630,1658,1667,1672,1795,1798,1804,1807,1849,1852,1857,1889,1894,1946,1949,1954,1973,1976,1981,2024,2027,2031,3104],[892,893,817],"p",{},[892,895,896,897,901,902,901,905,908],{},"总的来说组件之间有这三种关系：",[898,899,900],"strong",{},"父子关系","、",[898,903,904],{},"兄弟关系",[898,906,907],{},"隔代关系","（可能隔多代）。",[892,910,911],{},[912,913],"img",{"alt":11,"src":914},"\u002Fassets\u002Ffront-end\u002Fvue3\u002Fcomponent-communication\u002F2024-06-03-17-25-40.png",[892,916,917],{},[898,918,919,923,924,927],{},[920,921,922],"code",{},"Vue3","组件通信和",[920,925,926],{},"Vue2","的区别：",[929,930,931],"ul",{},[932,933,934,935,938],"li",{},"移出事件总线，使用",[920,936,937],{},"mitt","代替。",[929,940,941,951,962,972],{},[932,942,943,946,947,950],{},[920,944,945],{},"vuex","换成了",[920,948,949],{},"pinia","。",[932,952,953,954,957,958,961],{},"把",[920,955,956],{},".sync","优化到了",[920,959,960],{},"v-model","里面了。",[932,963,953,964,967,968,971],{},[920,965,966],{},"$listeners","所有的东西，合并到",[920,969,970],{},"$attrs","中了。",[932,973,974,977],{},[920,975,976],{},"$children","被砍掉了。",[892,979,980],{},[898,981,982],{},"常见搭配形式：",[892,984,985],{},[912,986],{"alt":11,"src":987},"\u002Fassets\u002Ffront-end\u002Fvue3\u002Fcomponent-communication\u002F2024-06-03-17-29-53.png",[989,990,992],"h2",{"id":991},"_1-props",[898,993,994],{},"1. props",[892,996,997],{},"Props 是使用频率最高的一种通信方式，通常被使用于 父组件向子组件传递数据。",[999,1000],"warning",{"title":1001},"理论上也能从子组件传递给父组件，但是需要父组件预先向子组件提供一个接受数据的函数。",[892,1003,1004],{},"父组件向子组件传递数据时，直接写在子组件的标签中。具体的写法如下所示：",[929,1006,1007],{},[932,1008,1009],{},"父组件传递数据",[1011,1012,1016],"pre",{"className":1013,"code":1014,"language":1015,"meta":11,"style":11},"language-vue shiki shiki-themes github-light github-light github-dark","\u003CChild v-bind:name=\"data\" \u002F>\n","vue",[920,1017,1018],{"__ignoreMap":11},[1019,1020,1023,1027,1031,1035,1038,1041,1044,1048,1051,1053],"span",{"class":1021,"line":1022},"line",1,[1019,1024,1026],{"class":1025},"sxrX7","\u003C",[1019,1028,1030],{"class":1029},"sovSZ","Child",[1019,1032,1034],{"class":1033},"snPdu"," v-bind",[1019,1036,1037],{"class":1025},":",[1019,1039,1040],{"class":1033},"name",[1019,1042,1043],{"class":1025},"=",[1019,1045,1047],{"class":1046},"sIIMD","\"",[1019,1049,1050],{"class":1025},"data",[1019,1052,1047],{"class":1046},[1019,1054,1055],{"class":1025}," \u002F>\n",[892,1057,1058,1059,1062,1063,1065,1066,1068],{},"传递时需要使用 ",[920,1060,1061],{},"v-bind"," 进行响应式绑定，子组件中拿到的数据名为 ",[920,1064,1040],{},"，实际的数据值为 ",[920,1067,1050],{}," 变量的值。",[892,1070,1071],{},"也可以直接传递方法，子组件接收后可直接调用。",[929,1073,1074],{},[932,1075,1076],{},"子组件接收数据",[1011,1078,1082],{"className":1079,"code":1080,"language":1081,"meta":11,"style":11},"language-ts shiki shiki-themes github-light github-light github-dark","defineProps(['car','getToy'])\n","ts",[920,1083,1084],{"__ignoreMap":11},[1019,1085,1086,1089,1092,1095,1098,1101],{"class":1021,"line":1022},[1019,1087,1088],{"class":1033},"defineProps",[1019,1090,1091],{"class":1025},"([",[1019,1093,1094],{"class":1046},"'car'",[1019,1096,1097],{"class":1025},",",[1019,1099,1100],{"class":1046},"'getToy'",[1019,1102,1103],{"class":1025},"])\n",[892,1105,1106,1107],{},"子组件需要先引入父组件传递的数据后，才能进行使用。\n",[999,1108],{"title":1109},"在 Vue3 中，defineProps 是一个内置函数，故不需要使用 import 引入，可直接在 setup 函数中使用。",[1111,1112,1114,1115,1411,1416],"note",{"title":1113},"Props 案例","\n- 父组件\n",[1011,1116,1118],{"className":1013,"code":1117,"language":1015,"meta":11,"style":11},"\u003Ctemplate>\n  \u003Cdiv class=\"father\">\n    \u003Ch3>父组件，\u003C\u002Fh3>\n        \u003Ch4>我的车：{{ car }}\u003C\u002Fh4>\n        \u003Ch4>儿子给的玩具：{{ toy }}\u003C\u002Fh4>\n        \u003CChild :car=\"car\" :getToy=\"getToy\"\u002F>\n  \u003C\u002Fdiv>\n\u003C\u002Ftemplate>\n\n\u003Cscript setup lang=\"ts\" name=\"Father\">\n    import Child from '.\u002FChild.vue'\n    import { ref } from \"vue\";\n    \u002F\u002F 数据\n    const car = ref('奔驰')\n    const toy = ref()\n    \u002F\u002F 方法\n    function getToy(value:string){\n        toy.value = value\n    }\n\u003C\u002Fscript>\n",[920,1119,1120,1130,1149,1165,1181,1195,1221,1231,1241,1248,1277,1293,1309,1316,1341,1356,1362,1385,1396,1402],{"__ignoreMap":11},[1019,1121,1122,1124,1127],{"class":1021,"line":1022},[1019,1123,1026],{"class":1025},[1019,1125,1126],{"class":1029},"template",[1019,1128,1129],{"class":1025},">\n",[1019,1131,1133,1136,1139,1142,1144,1147],{"class":1021,"line":1132},2,[1019,1134,1135],{"class":1025},"  \u003C",[1019,1137,1138],{"class":1029},"div",[1019,1140,1141],{"class":1033}," class",[1019,1143,1043],{"class":1025},[1019,1145,1146],{"class":1046},"\"father\"",[1019,1148,1129],{"class":1025},[1019,1150,1152,1155,1158,1161,1163],{"class":1021,"line":1151},3,[1019,1153,1154],{"class":1025},"    \u003C",[1019,1156,1157],{"class":1029},"h3",[1019,1159,1160],{"class":1025},">父组件，\u003C\u002F",[1019,1162,1157],{"class":1029},[1019,1164,1129],{"class":1025},[1019,1166,1168,1171,1174,1177,1179],{"class":1021,"line":1167},4,[1019,1169,1170],{"class":1025},"        \u003C",[1019,1172,1173],{"class":1029},"h4",[1019,1175,1176],{"class":1025},">我的车：{{ car }}\u003C\u002F",[1019,1178,1173],{"class":1029},[1019,1180,1129],{"class":1025},[1019,1182,1184,1186,1188,1191,1193],{"class":1021,"line":1183},5,[1019,1185,1170],{"class":1025},[1019,1187,1173],{"class":1029},[1019,1189,1190],{"class":1025},">儿子给的玩具：{{ toy }}\u003C\u002F",[1019,1192,1173],{"class":1029},[1019,1194,1129],{"class":1025},[1019,1196,1198,1200,1202,1205,1207,1210,1213,1215,1218],{"class":1021,"line":1197},6,[1019,1199,1170],{"class":1025},[1019,1201,1030],{"class":1029},[1019,1203,1204],{"class":1033}," :car",[1019,1206,1043],{"class":1025},[1019,1208,1209],{"class":1046},"\"car\"",[1019,1211,1212],{"class":1033}," :getToy",[1019,1214,1043],{"class":1025},[1019,1216,1217],{"class":1046},"\"getToy\"",[1019,1219,1220],{"class":1025},"\u002F>\n",[1019,1222,1224,1227,1229],{"class":1021,"line":1223},7,[1019,1225,1226],{"class":1025},"  \u003C\u002F",[1019,1228,1138],{"class":1029},[1019,1230,1129],{"class":1025},[1019,1232,1234,1237,1239],{"class":1021,"line":1233},8,[1019,1235,1236],{"class":1025},"\u003C\u002F",[1019,1238,1126],{"class":1029},[1019,1240,1129],{"class":1025},[1019,1242,1244],{"class":1021,"line":1243},9,[1019,1245,1247],{"emptyLinePlaceholder":1246},true,"\n",[1019,1249,1251,1253,1256,1259,1262,1264,1267,1270,1272,1275],{"class":1021,"line":1250},10,[1019,1252,1026],{"class":1025},[1019,1254,1255],{"class":1029},"script",[1019,1257,1258],{"class":1033}," setup",[1019,1260,1261],{"class":1033}," lang",[1019,1263,1043],{"class":1025},[1019,1265,1266],{"class":1046},"\"ts\"",[1019,1268,1269],{"class":1033}," name",[1019,1271,1043],{"class":1025},[1019,1273,1274],{"class":1046},"\"Father\"",[1019,1276,1129],{"class":1025},[1019,1278,1280,1284,1287,1290],{"class":1021,"line":1279},11,[1019,1281,1283],{"class":1282},"s8jYJ","    import",[1019,1285,1286],{"class":1025}," Child ",[1019,1288,1289],{"class":1282},"from",[1019,1291,1292],{"class":1046}," '.\u002FChild.vue'\n",[1019,1294,1296,1298,1301,1303,1306],{"class":1021,"line":1295},12,[1019,1297,1283],{"class":1282},[1019,1299,1300],{"class":1025}," { ref } ",[1019,1302,1289],{"class":1282},[1019,1304,1305],{"class":1046}," \"vue\"",[1019,1307,1308],{"class":1025},";\n",[1019,1310,1312],{"class":1021,"line":1311},13,[1019,1313,1315],{"class":1314},"sCsY4","    \u002F\u002F 数据\n",[1019,1317,1319,1322,1326,1329,1332,1335,1338],{"class":1021,"line":1318},14,[1019,1320,1321],{"class":1282},"    const",[1019,1323,1325],{"class":1324},"sBjJW"," car",[1019,1327,1328],{"class":1282}," =",[1019,1330,1331],{"class":1033}," ref",[1019,1333,1334],{"class":1025},"(",[1019,1336,1337],{"class":1046},"'奔驰'",[1019,1339,1340],{"class":1025},")\n",[1019,1342,1344,1346,1349,1351,1353],{"class":1021,"line":1343},15,[1019,1345,1321],{"class":1282},[1019,1347,1348],{"class":1324}," toy",[1019,1350,1328],{"class":1282},[1019,1352,1331],{"class":1033},[1019,1354,1355],{"class":1025},"()\n",[1019,1357,1359],{"class":1021,"line":1358},16,[1019,1360,1361],{"class":1314},"    \u002F\u002F 方法\n",[1019,1363,1365,1368,1371,1373,1377,1379,1382],{"class":1021,"line":1364},17,[1019,1366,1367],{"class":1282},"    function",[1019,1369,1370],{"class":1033}," getToy",[1019,1372,1334],{"class":1025},[1019,1374,1376],{"class":1375},"sP4rz","value",[1019,1378,1037],{"class":1282},[1019,1380,1381],{"class":1324},"string",[1019,1383,1384],{"class":1025},"){\n",[1019,1386,1388,1391,1393],{"class":1021,"line":1387},18,[1019,1389,1390],{"class":1025},"        toy.value ",[1019,1392,1043],{"class":1282},[1019,1394,1395],{"class":1025}," value\n",[1019,1397,1399],{"class":1021,"line":1398},19,[1019,1400,1401],{"class":1025},"    }\n",[1019,1403,1405,1407,1409],{"class":1021,"line":1404},20,[1019,1406,1236],{"class":1025},[1019,1408,1255],{"class":1029},[1019,1410,1129],{"class":1025},[929,1412,1413],{},[932,1414,1415],{},"子组件",[1011,1417,1419],{"className":1013,"code":1418,"language":1015,"meta":11,"style":11},"\u003Ctemplate>\n  \u003Cdiv class=\"child\">\n    \u003Ch3>子组件\u003C\u002Fh3>\n        \u003Ch4>我的玩具：{{ toy }}\u003C\u002Fh4>\n        \u003Ch4>父给我的车：{{ car }}\u003C\u002Fh4>\n        \u003Cbutton @click=\"getToy(toy)\">玩具给父亲\u003C\u002Fbutton>\n  \u003C\u002Fdiv>\n\u003C\u002Ftemplate>\n\n\u003Cscript setup lang=\"ts\" name=\"Child\">\n    import { ref } from \"vue\";\n    const toy = ref('奥特曼')\n\n    defineProps(['car','getToy'])\n\u003C\u002Fscript>\n",[920,1420,1421,1429,1444,1457,1470,1483,1505,1513,1521,1525,1548,1560,1577,1581,1596],{"__ignoreMap":11},[1019,1422,1423,1425,1427],{"class":1021,"line":1022},[1019,1424,1026],{"class":1025},[1019,1426,1126],{"class":1029},[1019,1428,1129],{"class":1025},[1019,1430,1431,1433,1435,1437,1439,1442],{"class":1021,"line":1132},[1019,1432,1135],{"class":1025},[1019,1434,1138],{"class":1029},[1019,1436,1141],{"class":1033},[1019,1438,1043],{"class":1025},[1019,1440,1441],{"class":1046},"\"child\"",[1019,1443,1129],{"class":1025},[1019,1445,1446,1448,1450,1453,1455],{"class":1021,"line":1151},[1019,1447,1154],{"class":1025},[1019,1449,1157],{"class":1029},[1019,1451,1452],{"class":1025},">子组件\u003C\u002F",[1019,1454,1157],{"class":1029},[1019,1456,1129],{"class":1025},[1019,1458,1459,1461,1463,1466,1468],{"class":1021,"line":1167},[1019,1460,1170],{"class":1025},[1019,1462,1173],{"class":1029},[1019,1464,1465],{"class":1025},">我的玩具：{{ toy }}\u003C\u002F",[1019,1467,1173],{"class":1029},[1019,1469,1129],{"class":1025},[1019,1471,1472,1474,1476,1479,1481],{"class":1021,"line":1183},[1019,1473,1170],{"class":1025},[1019,1475,1173],{"class":1029},[1019,1477,1478],{"class":1025},">父给我的车：{{ car }}\u003C\u002F",[1019,1480,1173],{"class":1029},[1019,1482,1129],{"class":1025},[1019,1484,1485,1487,1490,1493,1495,1498,1501,1503],{"class":1021,"line":1197},[1019,1486,1170],{"class":1025},[1019,1488,1489],{"class":1029},"button",[1019,1491,1492],{"class":1033}," @click",[1019,1494,1043],{"class":1025},[1019,1496,1497],{"class":1046},"\"getToy(toy)\"",[1019,1499,1500],{"class":1025},">玩具给父亲\u003C\u002F",[1019,1502,1489],{"class":1029},[1019,1504,1129],{"class":1025},[1019,1506,1507,1509,1511],{"class":1021,"line":1223},[1019,1508,1226],{"class":1025},[1019,1510,1138],{"class":1029},[1019,1512,1129],{"class":1025},[1019,1514,1515,1517,1519],{"class":1021,"line":1233},[1019,1516,1236],{"class":1025},[1019,1518,1126],{"class":1029},[1019,1520,1129],{"class":1025},[1019,1522,1523],{"class":1021,"line":1243},[1019,1524,1247],{"emptyLinePlaceholder":1246},[1019,1526,1527,1529,1531,1533,1535,1537,1539,1541,1543,1546],{"class":1021,"line":1250},[1019,1528,1026],{"class":1025},[1019,1530,1255],{"class":1029},[1019,1532,1258],{"class":1033},[1019,1534,1261],{"class":1033},[1019,1536,1043],{"class":1025},[1019,1538,1266],{"class":1046},[1019,1540,1269],{"class":1033},[1019,1542,1043],{"class":1025},[1019,1544,1545],{"class":1046},"\"Child\"",[1019,1547,1129],{"class":1025},[1019,1549,1550,1552,1554,1556,1558],{"class":1021,"line":1279},[1019,1551,1283],{"class":1282},[1019,1553,1300],{"class":1025},[1019,1555,1289],{"class":1282},[1019,1557,1305],{"class":1046},[1019,1559,1308],{"class":1025},[1019,1561,1562,1564,1566,1568,1570,1572,1575],{"class":1021,"line":1295},[1019,1563,1321],{"class":1282},[1019,1565,1348],{"class":1324},[1019,1567,1328],{"class":1282},[1019,1569,1331],{"class":1033},[1019,1571,1334],{"class":1025},[1019,1573,1574],{"class":1046},"'奥特曼'",[1019,1576,1340],{"class":1025},[1019,1578,1579],{"class":1021,"line":1311},[1019,1580,1247],{"emptyLinePlaceholder":1246},[1019,1582,1583,1586,1588,1590,1592,1594],{"class":1021,"line":1318},[1019,1584,1585],{"class":1033},"    defineProps",[1019,1587,1091],{"class":1025},[1019,1589,1094],{"class":1046},[1019,1591,1097],{"class":1025},[1019,1593,1100],{"class":1046},[1019,1595,1103],{"class":1025},[1019,1597,1598,1600,1602],{"class":1021,"line":1343},[1019,1599,1236],{"class":1025},[1019,1601,1255],{"class":1029},[1019,1603,1129],{"class":1025},[989,1605,1607],{"id":1606},"_2-自定义事件",[898,1608,1609],{},"2. 自定义事件",[892,1611,1612],{},"自定义事件通常用于 子组件 -> 父组件传递数据，这里需要注意的是自定义事件和自定义方法的区别，还要区分原生事件和自定义事件。",[929,1614,1615,1618],{},[932,1616,1617],{},"原生事件：事件名是特定的，事件对象是包含事件相关信息的对象。",[932,1619,1620],{},"自定义事件：事件名是任意名称，事件对象是调用 emit 是所提供的数据，可以是任意类型。",[999,1622],{"title":1623},"在自定义事件的命名中推荐使用短横线命名法，例如 get-user ，这是因为 HTML 中不区分大小写，大小驼峰命名法中存在的大写字母会被自动转换，导致歧义。",[1625,1626,1627],"ol",{},[932,1628,1629],{},"父组件首先给子组件绑定事件",[1011,1631,1633],{"className":1013,"code":1632,"language":1015,"meta":11,"style":11},"\u003CChild @send-toy=\"saveToy\" \u002F>\n",[920,1634,1635],{"__ignoreMap":11},[1019,1636,1637,1639,1641,1644,1647,1649,1651,1654,1656],{"class":1021,"line":1022},[1019,1638,1026],{"class":1025},[1019,1640,1030],{"class":1029},[1019,1642,1643],{"class":1025}," @",[1019,1645,1646],{"class":1033},"send-toy",[1019,1648,1043],{"class":1025},[1019,1650,1047],{"class":1046},[1019,1652,1653],{"class":1025},"saveToy",[1019,1655,1047],{"class":1046},[1019,1657,1055],{"class":1025},[892,1659,1660,1661,1663,1664,1666],{},"这里就给 ",[920,1662,1030],{}," 子组件绑定了 ",[920,1665,1646],{}," 事件。",[1625,1668,1669],{"start":1132},[932,1670,1671],{},"子组件中声明和触发事件",[1011,1673,1675],{"className":1013,"code":1674,"language":1015,"meta":11,"style":11},"\u003Ctemplate>\n    \u003C!-- 触发 send-toy 事件 -->\n    \u003Cbutton @click=\"emit('send-toy',toy)\">发送玩具\u003C\u002Fbutton>\n\u003C\u002Ftemplate>\n\n\n\u003Cscript setup lang=\"ts\">\nlet toy = ref(\"钢铁侠\");\n\u002F\u002F 声明事件\nconst emit = defineEmits(['send-toy'])\n\u003C\u002Fscript>\n",[920,1676,1677,1685,1690,1710,1718,1722,1726,1742,1762,1767,1787],{"__ignoreMap":11},[1019,1678,1679,1681,1683],{"class":1021,"line":1022},[1019,1680,1026],{"class":1025},[1019,1682,1126],{"class":1029},[1019,1684,1129],{"class":1025},[1019,1686,1687],{"class":1021,"line":1132},[1019,1688,1689],{"class":1314},"    \u003C!-- 触发 send-toy 事件 -->\n",[1019,1691,1692,1694,1696,1698,1700,1703,1706,1708],{"class":1021,"line":1151},[1019,1693,1154],{"class":1025},[1019,1695,1489],{"class":1029},[1019,1697,1492],{"class":1033},[1019,1699,1043],{"class":1025},[1019,1701,1702],{"class":1046},"\"emit('send-toy',toy)\"",[1019,1704,1705],{"class":1025},">发送玩具\u003C\u002F",[1019,1707,1489],{"class":1029},[1019,1709,1129],{"class":1025},[1019,1711,1712,1714,1716],{"class":1021,"line":1167},[1019,1713,1236],{"class":1025},[1019,1715,1126],{"class":1029},[1019,1717,1129],{"class":1025},[1019,1719,1720],{"class":1021,"line":1183},[1019,1721,1247],{"emptyLinePlaceholder":1246},[1019,1723,1724],{"class":1021,"line":1197},[1019,1725,1247],{"emptyLinePlaceholder":1246},[1019,1727,1728,1730,1732,1734,1736,1738,1740],{"class":1021,"line":1223},[1019,1729,1026],{"class":1025},[1019,1731,1255],{"class":1029},[1019,1733,1258],{"class":1033},[1019,1735,1261],{"class":1033},[1019,1737,1043],{"class":1025},[1019,1739,1266],{"class":1046},[1019,1741,1129],{"class":1025},[1019,1743,1744,1747,1750,1752,1754,1756,1759],{"class":1021,"line":1233},[1019,1745,1746],{"class":1282},"let",[1019,1748,1749],{"class":1025}," toy ",[1019,1751,1043],{"class":1282},[1019,1753,1331],{"class":1033},[1019,1755,1334],{"class":1025},[1019,1757,1758],{"class":1046},"\"钢铁侠\"",[1019,1760,1761],{"class":1025},");\n",[1019,1763,1764],{"class":1021,"line":1243},[1019,1765,1766],{"class":1314},"\u002F\u002F 声明事件\n",[1019,1768,1769,1772,1775,1777,1780,1782,1785],{"class":1021,"line":1250},[1019,1770,1771],{"class":1282},"const",[1019,1773,1774],{"class":1324}," emit",[1019,1776,1328],{"class":1282},[1019,1778,1779],{"class":1033}," defineEmits",[1019,1781,1091],{"class":1025},[1019,1783,1784],{"class":1046},"'send-toy'",[1019,1786,1103],{"class":1025},[1019,1788,1789,1791,1793],{"class":1021,"line":1279},[1019,1790,1236],{"class":1025},[1019,1792,1255],{"class":1029},[1019,1794,1129],{"class":1025},[1111,1796],{"title":1797},"子组件触发了父组件传递的事件后，父组件中对应的方法则会被调用，从而实现了子组件向父组件传递数据的效果。",[989,1799,1801],{"id":1800},"_3-mitt",[898,1802,1803],{},"3. mitt",[892,1805,1806],{},"mitt 与订阅发布模式类似，可以实现任意组件之间互相通信，在正式使用之前，请先安装 mitt",[1111,1808,1809,1810,1813,1831,1834],{},"\n@tab npm\n```bash\nnpm install mitt\n```\n",[892,1811,1812],{},"@tab pnpm",[1011,1814,1818],{"className":1815,"code":1816,"language":1817,"meta":11,"style":11},"language-bash shiki shiki-themes github-light github-light github-dark","pnpm install mitt\n","bash",[920,1819,1820],{"__ignoreMap":11},[1019,1821,1822,1825,1828],{"class":1021,"line":1022},[1019,1823,1824],{"class":1033},"pnpm",[1019,1826,1827],{"class":1046}," install",[1019,1829,1830],{"class":1046}," mitt\n",[892,1832,1833],{},"@tab yarn",[1011,1835,1837],{"className":1815,"code":1836,"language":1817,"meta":11,"style":11},"yarn add mitt\n",[920,1838,1839],{"__ignoreMap":11},[1019,1840,1841,1844,1847],{"class":1021,"line":1022},[1019,1842,1843],{"class":1033},"yarn",[1019,1845,1846],{"class":1046}," add",[1019,1848,1830],{"class":1046},[892,1850,1851],{},"mitt 使用步骤如下：",[1625,1853,1854],{},[932,1855,1856],{},"引入 mitt",[1011,1858,1860],{"className":1079,"code":1859,"language":1081,"meta":11,"style":11},"import mitt from 'mitt'\nconst emitter = mitt()\n",[920,1861,1862,1875],{"__ignoreMap":11},[1019,1863,1864,1867,1870,1872],{"class":1021,"line":1022},[1019,1865,1866],{"class":1282},"import",[1019,1868,1869],{"class":1025}," mitt ",[1019,1871,1289],{"class":1282},[1019,1873,1874],{"class":1046}," 'mitt'\n",[1019,1876,1877,1879,1882,1884,1887],{"class":1021,"line":1132},[1019,1878,1771],{"class":1282},[1019,1880,1881],{"class":1324}," emitter",[1019,1883,1328],{"class":1282},[1019,1885,1886],{"class":1033}," mitt",[1019,1888,1355],{"class":1025},[1625,1890,1891],{"start":1132},[932,1892,1893],{},"配置事件绑定",[1011,1895,1897],{"className":1079,"code":1896,"language":1081,"meta":11,"style":11},"emitter.on('send-toy',(value : any)=>{\n    console.log(value);\n})\n",[920,1898,1899,1931,1941],{"__ignoreMap":11},[1019,1900,1901,1904,1907,1909,1911,1914,1916,1919,1922,1925,1928],{"class":1021,"line":1022},[1019,1902,1903],{"class":1025},"emitter.",[1019,1905,1906],{"class":1033},"on",[1019,1908,1334],{"class":1025},[1019,1910,1784],{"class":1046},[1019,1912,1913],{"class":1025},",(",[1019,1915,1376],{"class":1375},[1019,1917,1918],{"class":1282}," :",[1019,1920,1921],{"class":1324}," any",[1019,1923,1924],{"class":1025},")",[1019,1926,1927],{"class":1282},"=>",[1019,1929,1930],{"class":1025},"{\n",[1019,1932,1933,1936,1938],{"class":1021,"line":1132},[1019,1934,1935],{"class":1025},"    console.",[1019,1937,260],{"class":1033},[1019,1939,1940],{"class":1025},"(value);\n",[1019,1942,1943],{"class":1021,"line":1151},[1019,1944,1945],{"class":1025},"})\n",[892,1947,1948],{},"emitter 的 on 方法有两个参数，第一个参数为自定义事件的名称，第二个为事件触发时的回调函数。",[1625,1950,1951],{"start":1151},[932,1952,1953],{},"事件调用",[1011,1955,1957],{"className":1079,"code":1956,"language":1081,"meta":11,"style":11},"emitter.emit('send-toy',toy.value)\n",[920,1958,1959],{"__ignoreMap":11},[1019,1960,1961,1963,1966,1968,1970],{"class":1021,"line":1022},[1019,1962,1903],{"class":1025},[1019,1964,1965],{"class":1033},"emit",[1019,1967,1334],{"class":1025},[1019,1969,1784],{"class":1046},[1019,1971,1972],{"class":1025},",toy.value)\n",[892,1974,1975],{},"emitter 的 emit 方法来触发事件，第一个参数为绑定的自定义事件名称，后续参数是回调函数的参数列表。",[1625,1977,1978],{"start":1167},[932,1979,1980],{},"事件解绑",[1011,1982,1984],{"className":1079,"code":1983,"language":1081,"meta":11,"style":11},"\u002F\u002F 解绑单个事件\nemitter.off('send-toy');\n\n\u002F\u002F 解绑所有事件\nemitter.all.clear();\n",[920,1985,1986,1991,2004,2008,2013],{"__ignoreMap":11},[1019,1987,1988],{"class":1021,"line":1022},[1019,1989,1990],{"class":1314},"\u002F\u002F 解绑单个事件\n",[1019,1992,1993,1995,1998,2000,2002],{"class":1021,"line":1132},[1019,1994,1903],{"class":1025},[1019,1996,1997],{"class":1033},"off",[1019,1999,1334],{"class":1025},[1019,2001,1784],{"class":1046},[1019,2003,1761],{"class":1025},[1019,2005,2006],{"class":1021,"line":1151},[1019,2007,1247],{"emptyLinePlaceholder":1246},[1019,2009,2010],{"class":1021,"line":1167},[1019,2011,2012],{"class":1314},"\u002F\u002F 解绑所有事件\n",[1019,2014,2015,2018,2021],{"class":1021,"line":1183},[1019,2016,2017],{"class":1025},"emitter.all.",[1019,2019,2020],{"class":1033},"clear",[1019,2022,2023],{"class":1025},"();\n",[892,2025,2026],{},"emitter 的 off 方法来触发事件，传递的参数为绑定的自定义事件名称。",[999,2028,2030],{"title":2029},"总结","\n本质上可以将 mitt 看做是将原来写在组件内部的自定义事件抽离出来，将其管理在 mitt 中，所有组件都通过 mitt 来实现自定义事件，从而实现任意组件之间的交互。类似于设计模式中的代理模式。\n",[1111,2032,2034,2035,2096,2101,2202,2207,2223,2226,2232,2240,2243,2248,2329,2334,2409,2413,2418,2422,2598,2601,2673,2679,2684,2689,2701,2704,2709,2860,2865,2963,2968,3005,3008,3015,3089,3093,3097,3100],{"title":2033},"mitt 案例","\n1. 新建 `\u002Fsrc\u002Futils\u002Femitter.ts` 文件\n",[1011,2036,2038],{"className":1079,"code":2037,"language":1081,"meta":11,"style":11},"\u002F\u002F 1. 引入 mitt\nimport mitt from 'mitt'\n\n\u002F\u002F 2. 创建 emitter \nconst emitter = mitt()\n\n\u002F\u002F 3. 暴露 mitt \nexport default emitter\n",[920,2039,2040,2045,2055,2059,2064,2076,2080,2085],{"__ignoreMap":11},[1019,2041,2042],{"class":1021,"line":1022},[1019,2043,2044],{"class":1314},"\u002F\u002F 1. 引入 mitt\n",[1019,2046,2047,2049,2051,2053],{"class":1021,"line":1132},[1019,2048,1866],{"class":1282},[1019,2050,1869],{"class":1025},[1019,2052,1289],{"class":1282},[1019,2054,1874],{"class":1046},[1019,2056,2057],{"class":1021,"line":1151},[1019,2058,1247],{"emptyLinePlaceholder":1246},[1019,2060,2061],{"class":1021,"line":1167},[1019,2062,2063],{"class":1314},"\u002F\u002F 2. 创建 emitter \n",[1019,2065,2066,2068,2070,2072,2074],{"class":1021,"line":1183},[1019,2067,1771],{"class":1282},[1019,2069,1881],{"class":1324},[1019,2071,1328],{"class":1282},[1019,2073,1886],{"class":1033},[1019,2075,1355],{"class":1025},[1019,2077,2078],{"class":1021,"line":1197},[1019,2079,1247],{"emptyLinePlaceholder":1246},[1019,2081,2082],{"class":1021,"line":1223},[1019,2083,2084],{"class":1314},"\u002F\u002F 3. 暴露 mitt \n",[1019,2086,2087,2090,2093],{"class":1021,"line":1233},[1019,2088,2089],{"class":1282},"export",[1019,2091,2092],{"class":1282}," default",[1019,2094,2095],{"class":1025}," emitter\n",[1625,2097,2098],{"start":1132},[932,2099,2100],{},"使用 mitt 来绑定一些事件",[1011,2102,2104],{"className":1079,"code":2103,"language":1081,"meta":11,"style":11},"import emitter from '@\u002Futils\u002Femitter'\n\n\u002F\u002F 绑定事件\nemitter.on('send-toy',(value)=>{\n    console.log('send-toy事件被触发',value)\n})\n\n\u002F\u002F 解绑事件\nemitter.off('send-toy');\n\n\u002F\u002F 解绑所有事件\nemitter.all.clear();\n",[920,2105,2106,2118,2122,2127,2147,2161,2165,2169,2174,2186,2190,2194],{"__ignoreMap":11},[1019,2107,2108,2110,2113,2115],{"class":1021,"line":1022},[1019,2109,1866],{"class":1282},[1019,2111,2112],{"class":1025}," emitter ",[1019,2114,1289],{"class":1282},[1019,2116,2117],{"class":1046}," '@\u002Futils\u002Femitter'\n",[1019,2119,2120],{"class":1021,"line":1132},[1019,2121,1247],{"emptyLinePlaceholder":1246},[1019,2123,2124],{"class":1021,"line":1151},[1019,2125,2126],{"class":1314},"\u002F\u002F 绑定事件\n",[1019,2128,2129,2131,2133,2135,2137,2139,2141,2143,2145],{"class":1021,"line":1167},[1019,2130,1903],{"class":1025},[1019,2132,1906],{"class":1033},[1019,2134,1334],{"class":1025},[1019,2136,1784],{"class":1046},[1019,2138,1913],{"class":1025},[1019,2140,1376],{"class":1375},[1019,2142,1924],{"class":1025},[1019,2144,1927],{"class":1282},[1019,2146,1930],{"class":1025},[1019,2148,2149,2151,2153,2155,2158],{"class":1021,"line":1183},[1019,2150,1935],{"class":1025},[1019,2152,260],{"class":1033},[1019,2154,1334],{"class":1025},[1019,2156,2157],{"class":1046},"'send-toy事件被触发'",[1019,2159,2160],{"class":1025},",value)\n",[1019,2162,2163],{"class":1021,"line":1197},[1019,2164,1945],{"class":1025},[1019,2166,2167],{"class":1021,"line":1223},[1019,2168,1247],{"emptyLinePlaceholder":1246},[1019,2170,2171],{"class":1021,"line":1233},[1019,2172,2173],{"class":1314},"\u002F\u002F 解绑事件\n",[1019,2175,2176,2178,2180,2182,2184],{"class":1021,"line":1243},[1019,2177,1903],{"class":1025},[1019,2179,1997],{"class":1033},[1019,2181,1334],{"class":1025},[1019,2183,1784],{"class":1046},[1019,2185,1761],{"class":1025},[1019,2187,2188],{"class":1021,"line":1250},[1019,2189,1247],{"emptyLinePlaceholder":1246},[1019,2191,2192],{"class":1021,"line":1279},[1019,2193,2012],{"class":1314},[1019,2195,2196,2198,2200],{"class":1021,"line":1295},[1019,2197,2017],{"class":1025},[1019,2199,2020],{"class":1033},[1019,2201,2023],{"class":1025},[1625,2203,2204],{"start":1151},[932,2205,2206],{},"触发事件",[1011,2208,2209],{"className":1079,"code":1956,"language":1081,"meta":11,"style":11},[920,2210,2211],{"__ignoreMap":11},[1019,2212,2213,2215,2217,2219,2221],{"class":1021,"line":1022},[1019,2214,1903],{"class":1025},[1019,2216,1965],{"class":1033},[1019,2218,1334],{"class":1025},[1019,2220,1784],{"class":1046},[1019,2222,1972],{"class":1025},[892,2224,2225],{},"> 在触发事件之前不能解绑事件，否则无法成功触发。\n",[989,2227,2229],{"id":2228},"_4-v-model",[898,2230,2231],{},"4. v-model",[892,2233,2234,2235,2239],{},"实际上在前面的 ",[2236,2237,840],"a",{"href":2238},"\u002Ffront-end\u002Fvue3\u002Ftemplate-grammar.html"," 中已经介绍过 v-model 了，回顾一下 v-model 用于数据的双向绑定。",[892,2241,2242],{},"由于是在模板语法章节，故没有过多的详细说明 v-model 的使用，这里来详细说明。",[1625,2244,2245],{},[932,2246,2247],{},"作用于普通标签",[1011,2249,2251],{"className":1013,"code":2250,"language":1015,"meta":11,"style":11},"\u003Ctemplate>\n    \u003Cinput v-model=\"username\" \u002F>\n\n    \u003C!-- 等价于 -->\n    \u003Cinput v-bind:value=\"username\"  \n           @input=\"username =($event.target as HTMLInputElement).value\" \n    \u002F>\n\u003C\u002Ftemplate>\n",[920,2252,2253,2261,2278,2282,2287,2303,2316,2321],{"__ignoreMap":11},[1019,2254,2255,2257,2259],{"class":1021,"line":1022},[1019,2256,1026],{"class":1025},[1019,2258,1126],{"class":1029},[1019,2260,1129],{"class":1025},[1019,2262,2263,2265,2268,2271,2273,2276],{"class":1021,"line":1132},[1019,2264,1154],{"class":1025},[1019,2266,2267],{"class":1029},"input",[1019,2269,2270],{"class":1033}," v-model",[1019,2272,1043],{"class":1025},[1019,2274,2275],{"class":1046},"\"username\"",[1019,2277,1055],{"class":1025},[1019,2279,2280],{"class":1021,"line":1151},[1019,2281,1247],{"emptyLinePlaceholder":1246},[1019,2283,2284],{"class":1021,"line":1167},[1019,2285,2286],{"class":1314},"    \u003C!-- 等价于 -->\n",[1019,2288,2289,2291,2293,2296,2298,2300],{"class":1021,"line":1183},[1019,2290,1154],{"class":1025},[1019,2292,2267],{"class":1029},[1019,2294,2295],{"class":1033}," v-bind:value",[1019,2297,1043],{"class":1025},[1019,2299,2275],{"class":1046},[1019,2301,2302],{"class":1025},"  \n",[1019,2304,2305,2308,2310,2313],{"class":1021,"line":1197},[1019,2306,2307],{"class":1033},"           @input",[1019,2309,1043],{"class":1025},[1019,2311,2312],{"class":1046},"\"username =($event.target as HTMLInputElement).value\"",[1019,2314,2315],{"class":1025}," \n",[1019,2317,2318],{"class":1021,"line":1223},[1019,2319,2320],{"class":1025},"    \u002F>\n",[1019,2322,2323,2325,2327],{"class":1021,"line":1233},[1019,2324,1236],{"class":1025},[1019,2326,1126],{"class":1029},[1019,2328,1129],{"class":1025},[1625,2330,2331],{"start":1132},[932,2332,2333],{},"作用于组件标签",[1011,2335,2337],{"className":1013,"code":2336,"language":1015,"meta":11,"style":11},"\u003Ctemplate>\n    \u003CMyInput v-model=\"username\" \u002F>\n\n    \u003C!-- 等价于 -->\n    \u003CMyInput :modelValue=\"username\"  \n            @update:modelValue=\"username = $event\" \n    \u002F>\n\u003C\u002Ftemplate>\n",[920,2338,2339,2347,2362,2366,2370,2385,2397,2401],{"__ignoreMap":11},[1019,2340,2341,2343,2345],{"class":1021,"line":1022},[1019,2342,1026],{"class":1025},[1019,2344,1126],{"class":1029},[1019,2346,1129],{"class":1025},[1019,2348,2349,2351,2354,2356,2358,2360],{"class":1021,"line":1132},[1019,2350,1154],{"class":1025},[1019,2352,2353],{"class":1029},"MyInput",[1019,2355,2270],{"class":1033},[1019,2357,1043],{"class":1025},[1019,2359,2275],{"class":1046},[1019,2361,1055],{"class":1025},[1019,2363,2364],{"class":1021,"line":1151},[1019,2365,1247],{"emptyLinePlaceholder":1246},[1019,2367,2368],{"class":1021,"line":1167},[1019,2369,2286],{"class":1314},[1019,2371,2372,2374,2376,2379,2381,2383],{"class":1021,"line":1183},[1019,2373,1154],{"class":1025},[1019,2375,2353],{"class":1029},[1019,2377,2378],{"class":1033}," :modelValue",[1019,2380,1043],{"class":1025},[1019,2382,2275],{"class":1046},[1019,2384,2302],{"class":1025},[1019,2386,2387,2390,2392,2395],{"class":1021,"line":1197},[1019,2388,2389],{"class":1033},"            @update:modelValue",[1019,2391,1043],{"class":1025},[1019,2393,2394],{"class":1046},"\"username = $event\"",[1019,2396,2315],{"class":1025},[1019,2398,2399],{"class":1021,"line":1223},[1019,2400,2320],{"class":1025},[1019,2402,2403,2405,2407],{"class":1021,"line":1233},[1019,2404,1236],{"class":1025},[1019,2406,1126],{"class":1029},[1019,2408,1129],{"class":1025},[999,2410,2412],{"title":2411},"$event 什么时候调用 target？","\n1. 当 $event 使用在原生 HTML 标签上时，代表的是 DOM 元素的事件对象，则需要调用 target \n2. 当 $event 使用在组件标签上时，即自定义事件，代表的就是事件触发时，所传递的数据，不需要调用 target\n",[1625,2414,2415],{"start":1151},[932,2416,2417],{},"这通常使用在自定义组件的封装中",[929,2419,2420],{},[932,2421,2353],{},[1011,2423,2425],{"className":1013,"code":2424,"language":1015,"meta":11,"style":11},"\u003Ctemplate>\n  \u003Cdiv class=\"box\">\n    \u003C!--将接收的value值赋给input元素的value属性，目的是：为了呈现数据 -->\n        \u003C!--给input元素绑定原生input事件，触发input事件时，进而触发update:model-value事件-->\n    \u003Cinput \n       type=\"text\" \n       :value=\"modelValue\" \n       @input=\"emit('update:model-value',$event.target.value)\"\n    >\n  \u003C\u002Fdiv>\n\u003C\u002Ftemplate>\n\n\u003Cscript setup lang=\"ts\" name=\"MyInput\">\n  \u002F\u002F 接收props\n  defineProps(['modelValue'])\n  \u002F\u002F 声明事件\n  const emit = defineEmits(['update:model-value'])\n\u003C\u002Fscript>\n",[920,2426,2427,2435,2450,2455,2460,2468,2480,2492,2502,2507,2515,2523,2527,2550,2555,2567,2572,2590],{"__ignoreMap":11},[1019,2428,2429,2431,2433],{"class":1021,"line":1022},[1019,2430,1026],{"class":1025},[1019,2432,1126],{"class":1029},[1019,2434,1129],{"class":1025},[1019,2436,2437,2439,2441,2443,2445,2448],{"class":1021,"line":1132},[1019,2438,1135],{"class":1025},[1019,2440,1138],{"class":1029},[1019,2442,1141],{"class":1033},[1019,2444,1043],{"class":1025},[1019,2446,2447],{"class":1046},"\"box\"",[1019,2449,1129],{"class":1025},[1019,2451,2452],{"class":1021,"line":1151},[1019,2453,2454],{"class":1314},"    \u003C!--将接收的value值赋给input元素的value属性，目的是：为了呈现数据 -->\n",[1019,2456,2457],{"class":1021,"line":1167},[1019,2458,2459],{"class":1314},"        \u003C!--给input元素绑定原生input事件，触发input事件时，进而触发update:model-value事件-->\n",[1019,2461,2462,2464,2466],{"class":1021,"line":1183},[1019,2463,1154],{"class":1025},[1019,2465,2267],{"class":1029},[1019,2467,2315],{"class":1025},[1019,2469,2470,2473,2475,2478],{"class":1021,"line":1197},[1019,2471,2472],{"class":1033},"       type",[1019,2474,1043],{"class":1025},[1019,2476,2477],{"class":1046},"\"text\"",[1019,2479,2315],{"class":1025},[1019,2481,2482,2485,2487,2490],{"class":1021,"line":1223},[1019,2483,2484],{"class":1033},"       :value",[1019,2486,1043],{"class":1025},[1019,2488,2489],{"class":1046},"\"modelValue\"",[1019,2491,2315],{"class":1025},[1019,2493,2494,2497,2499],{"class":1021,"line":1233},[1019,2495,2496],{"class":1033},"       @input",[1019,2498,1043],{"class":1025},[1019,2500,2501],{"class":1046},"\"emit('update:model-value',$event.target.value)\"\n",[1019,2503,2504],{"class":1021,"line":1243},[1019,2505,2506],{"class":1025},"    >\n",[1019,2508,2509,2511,2513],{"class":1021,"line":1250},[1019,2510,1226],{"class":1025},[1019,2512,1138],{"class":1029},[1019,2514,1129],{"class":1025},[1019,2516,2517,2519,2521],{"class":1021,"line":1279},[1019,2518,1236],{"class":1025},[1019,2520,1126],{"class":1029},[1019,2522,1129],{"class":1025},[1019,2524,2525],{"class":1021,"line":1295},[1019,2526,1247],{"emptyLinePlaceholder":1246},[1019,2528,2529,2531,2533,2535,2537,2539,2541,2543,2545,2548],{"class":1021,"line":1311},[1019,2530,1026],{"class":1025},[1019,2532,1255],{"class":1029},[1019,2534,1258],{"class":1033},[1019,2536,1261],{"class":1033},[1019,2538,1043],{"class":1025},[1019,2540,1266],{"class":1046},[1019,2542,1269],{"class":1033},[1019,2544,1043],{"class":1025},[1019,2546,2547],{"class":1046},"\"MyInput\"",[1019,2549,1129],{"class":1025},[1019,2551,2552],{"class":1021,"line":1318},[1019,2553,2554],{"class":1314},"  \u002F\u002F 接收props\n",[1019,2556,2557,2560,2562,2565],{"class":1021,"line":1343},[1019,2558,2559],{"class":1033},"  defineProps",[1019,2561,1091],{"class":1025},[1019,2563,2564],{"class":1046},"'modelValue'",[1019,2566,1103],{"class":1025},[1019,2568,2569],{"class":1021,"line":1358},[1019,2570,2571],{"class":1314},"  \u002F\u002F 声明事件\n",[1019,2573,2574,2577,2579,2581,2583,2585,2588],{"class":1021,"line":1364},[1019,2575,2576],{"class":1282},"  const",[1019,2578,1774],{"class":1324},[1019,2580,1328],{"class":1282},[1019,2582,1779],{"class":1033},[1019,2584,1091],{"class":1025},[1019,2586,2587],{"class":1046},"'update:model-value'",[1019,2589,1103],{"class":1025},[1019,2591,2592,2594,2596],{"class":1021,"line":1387},[1019,2593,1236],{"class":1025},[1019,2595,1255],{"class":1029},[1019,2597,1129],{"class":1025},[892,2599,2600],{},"父组件既向子组件传递了数据，子组件也向父组件传递了数据。",[1111,2602,2604,2605,2629],{"title":2603},"自定义 v-model 值","\n在子组件中接受 v-model 传递的值时，默认是使用的 modelValue，可以进行自定义。\n",[1625,2606,2607,2613,2623,2626],{},[932,2608,2609,2610],{},"传递时: ",[920,2611,2612],{},"v-model:username",[932,2614,2615,2616,2619,2620],{},"接收时：",[920,2617,2618],{},"defineProps(['username'])"," 和 ",[920,2621,2622],{},"defineEmits(['update:username'])",[932,2624,2625],{},"在模板中使用即可",[932,2627,2628],{},"当使用了自定义 v-model 值后，则可以在组件标签上多次使用 v-model",[1011,2630,2632],{"className":1013,"code":2631,"language":1015,"meta":11,"style":11},"\u003CChild v-model:username=\"username\" v-model:password=\"password\" ... \u002F>\n",[920,2633,2634],{"__ignoreMap":11},[1019,2635,2636,2638,2640,2642,2644,2647,2649,2651,2653,2655,2657,2659,2662,2664,2666,2668,2670],{"class":1021,"line":1022},[1019,2637,1026],{"class":1025},[1019,2639,1030],{"class":1029},[1019,2641,2270],{"class":1033},[1019,2643,1037],{"class":1025},[1019,2645,2646],{"class":1033},"username",[1019,2648,1043],{"class":1025},[1019,2650,1047],{"class":1046},[1019,2652,2646],{"class":1025},[1019,2654,1047],{"class":1046},[1019,2656,2270],{"class":1033},[1019,2658,1037],{"class":1025},[1019,2660,2661],{"class":1033},"password",[1019,2663,1043],{"class":1025},[1019,2665,1047],{"class":1046},[1019,2667,2661],{"class":1025},[1019,2669,1047],{"class":1046},[1019,2671,2672],{"class":1025}," ... \u002F>\n",[989,2674,2676],{"id":2675},"_5-attrs",[898,2677,2678],{},"5. $attrs",[892,2680,2681,2683],{},[920,2682,970],{}," 用于实现当前组件向其子组件传递数据通信的一种方式，也可跨代进行通信。祖组件 => 子组件",[892,2685,2686,2688],{},[920,2687,970],{},"是一个对象，包含所有父组件传入的标签属性。",[892,2690,2691,2693,2694,2697,2698,2700],{},[920,2692,970],{}," 会自动排除在组件传递过程中 ",[920,2695,2696],{},"props"," 中声明的属性，可以认为声明过的组件 ",[920,2699,2696],{}," 被子组件消费了，故不会再继续向下传递。",[892,2702,2703],{},"具体的使用方式：",[1625,2705,2706],{},[932,2707,2708],{},"父组件中向子组件绑定数据",[1011,2710,2712],{"className":1013,"code":2711,"language":1015,"meta":11,"style":11},"\u003Ctemplate>\n    \u003CChild v-bind:a=\"a\" :b=\"b\" :=\"{x:100,y:200}\" :addA=\"addA\">\n\u003C\u002Ftemplate>\n\u003Cscript>\nlet a = ref(0);\nlet b = ref(3);\n\nfunction addA(value : number){\n    a.value += value;\n}\n\u003C\u002Fscript>\n",[920,2713,2714,2722,2761,2769,2777,2795,2813,2817,2836,2847,2852],{"__ignoreMap":11},[1019,2715,2716,2718,2720],{"class":1021,"line":1022},[1019,2717,1026],{"class":1025},[1019,2719,1126],{"class":1029},[1019,2721,1129],{"class":1025},[1019,2723,2724,2726,2728,2731,2733,2736,2739,2741,2744,2746,2748,2751,2754,2756,2759],{"class":1021,"line":1132},[1019,2725,1154],{"class":1025},[1019,2727,1030],{"class":1029},[1019,2729,2730],{"class":1033}," v-bind:a",[1019,2732,1043],{"class":1025},[1019,2734,2735],{"class":1046},"\"a\"",[1019,2737,2738],{"class":1033}," :b",[1019,2740,1043],{"class":1025},[1019,2742,2743],{"class":1046},"\"b\"",[1019,2745,1918],{"class":1033},[1019,2747,1043],{"class":1025},[1019,2749,2750],{"class":1046},"\"{x:100,y:200}\"",[1019,2752,2753],{"class":1033}," :addA",[1019,2755,1043],{"class":1025},[1019,2757,2758],{"class":1046},"\"addA\"",[1019,2760,1129],{"class":1025},[1019,2762,2763,2765,2767],{"class":1021,"line":1151},[1019,2764,1236],{"class":1025},[1019,2766,1126],{"class":1029},[1019,2768,1129],{"class":1025},[1019,2770,2771,2773,2775],{"class":1021,"line":1167},[1019,2772,1026],{"class":1025},[1019,2774,1255],{"class":1029},[1019,2776,1129],{"class":1025},[1019,2778,2779,2781,2784,2786,2788,2790,2793],{"class":1021,"line":1183},[1019,2780,1746],{"class":1282},[1019,2782,2783],{"class":1025}," a ",[1019,2785,1043],{"class":1282},[1019,2787,1331],{"class":1033},[1019,2789,1334],{"class":1025},[1019,2791,2792],{"class":1324},"0",[1019,2794,1761],{"class":1025},[1019,2796,2797,2799,2802,2804,2806,2808,2811],{"class":1021,"line":1197},[1019,2798,1746],{"class":1282},[1019,2800,2801],{"class":1025}," b ",[1019,2803,1043],{"class":1282},[1019,2805,1331],{"class":1033},[1019,2807,1334],{"class":1025},[1019,2809,2810],{"class":1324},"3",[1019,2812,1761],{"class":1025},[1019,2814,2815],{"class":1021,"line":1223},[1019,2816,1247],{"emptyLinePlaceholder":1246},[1019,2818,2819,2822,2825,2827,2829,2831,2834],{"class":1021,"line":1233},[1019,2820,2821],{"class":1282},"function",[1019,2823,2824],{"class":1033}," addA",[1019,2826,1334],{"class":1025},[1019,2828,1376],{"class":1375},[1019,2830,1918],{"class":1282},[1019,2832,2833],{"class":1324}," number",[1019,2835,1384],{"class":1025},[1019,2837,2838,2841,2844],{"class":1021,"line":1243},[1019,2839,2840],{"class":1025},"    a.value ",[1019,2842,2843],{"class":1282},"+=",[1019,2845,2846],{"class":1025}," value;\n",[1019,2848,2849],{"class":1021,"line":1250},[1019,2850,2851],{"class":1025},"}\n",[1019,2853,2854,2856,2858],{"class":1021,"line":1279},[1019,2855,1236],{"class":1025},[1019,2857,1255],{"class":1029},[1019,2859,1129],{"class":1025},[1625,2861,2862],{"start":1132},[932,2863,2864],{},"子组件消费数据",[1011,2866,2868],{"className":1013,"code":2867,"language":1015,"meta":11,"style":11},"\u003Ctemplate>\n    \u003Ch4>a：{{ a }}\u003C\u002Fh4>\n    \u003Cbutton @click=\"addA(3)\">点我更新 A\u003C\u002Fbutton>\n\u003C\u002Ftemplate>\n\n\u003Cscript setup lang=\"ts\">\ndefineProps(['a','addA'])\n\u003C\u002Fscript>\n",[920,2869,2870,2878,2891,2911,2919,2923,2939,2955],{"__ignoreMap":11},[1019,2871,2872,2874,2876],{"class":1021,"line":1022},[1019,2873,1026],{"class":1025},[1019,2875,1126],{"class":1029},[1019,2877,1129],{"class":1025},[1019,2879,2880,2882,2884,2887,2889],{"class":1021,"line":1132},[1019,2881,1154],{"class":1025},[1019,2883,1173],{"class":1029},[1019,2885,2886],{"class":1025},">a：{{ a }}\u003C\u002F",[1019,2888,1173],{"class":1029},[1019,2890,1129],{"class":1025},[1019,2892,2893,2895,2897,2899,2901,2904,2907,2909],{"class":1021,"line":1151},[1019,2894,1154],{"class":1025},[1019,2896,1489],{"class":1029},[1019,2898,1492],{"class":1033},[1019,2900,1043],{"class":1025},[1019,2902,2903],{"class":1046},"\"addA(3)\"",[1019,2905,2906],{"class":1025},">点我更新 A\u003C\u002F",[1019,2908,1489],{"class":1029},[1019,2910,1129],{"class":1025},[1019,2912,2913,2915,2917],{"class":1021,"line":1167},[1019,2914,1236],{"class":1025},[1019,2916,1126],{"class":1029},[1019,2918,1129],{"class":1025},[1019,2920,2921],{"class":1021,"line":1183},[1019,2922,1247],{"emptyLinePlaceholder":1246},[1019,2924,2925,2927,2929,2931,2933,2935,2937],{"class":1021,"line":1197},[1019,2926,1026],{"class":1025},[1019,2928,1255],{"class":1029},[1019,2930,1258],{"class":1033},[1019,2932,1261],{"class":1033},[1019,2934,1043],{"class":1025},[1019,2936,1266],{"class":1046},[1019,2938,1129],{"class":1025},[1019,2940,2941,2943,2945,2948,2950,2953],{"class":1021,"line":1223},[1019,2942,1088],{"class":1033},[1019,2944,1091],{"class":1025},[1019,2946,2947],{"class":1046},"'a'",[1019,2949,1097],{"class":1025},[1019,2951,2952],{"class":1046},"'addA'",[1019,2954,1103],{"class":1025},[1019,2956,2957,2959,2961],{"class":1021,"line":1233},[1019,2958,1236],{"class":1025},[1019,2960,1255],{"class":1029},[1019,2962,1129],{"class":1025},[1625,2964,2965],{"start":1151},[932,2966,2967],{},"子组件向孙组件传递数据",[1011,2969,2971],{"className":1013,"code":2970,"language":1015,"meta":11,"style":11},"\u003Ctemplate>\n    \u003CGrantChild v-bind=\"$attrs\" \u002F>\n\u003C\u002Ftemplate>\n",[920,2972,2973,2981,2997],{"__ignoreMap":11},[1019,2974,2975,2977,2979],{"class":1021,"line":1022},[1019,2976,1026],{"class":1025},[1019,2978,1126],{"class":1029},[1019,2980,1129],{"class":1025},[1019,2982,2983,2985,2988,2990,2992,2995],{"class":1021,"line":1132},[1019,2984,1154],{"class":1025},[1019,2986,2987],{"class":1029},"GrantChild",[1019,2989,1034],{"class":1033},[1019,2991,1043],{"class":1025},[1019,2993,2994],{"class":1046},"\"$attrs\"",[1019,2996,1055],{"class":1025},[1019,2998,2999,3001,3003],{"class":1021,"line":1151},[1019,3000,1236],{"class":1025},[1019,3002,1126],{"class":1029},[1019,3004,1129],{"class":1025},[1111,3006],{"title":3007},"此时，孙组件中能使用的数据就剩下了 b、x、y 这三个数据，a 和 addA 被子组件消费了，不会向后传递。",[989,3009,3011,3012],{"id":3010},"_6-refsparent","6. ",[898,3013,3014],{},"$refs、$parent",[1625,3016,3017,3040],{},[932,3018,3019,3020],{},"概述：",[929,3021,3022,3031],{},[932,3023,3024,3027,3028],{},[920,3025,3026],{},"$refs","用于 ：",[898,3029,3030],{},"父→子。",[932,3032,3033,3036,3037],{},[920,3034,3035],{},"$parent","用于：",[898,3038,3039],{},"子→父。",[932,3041,3042,3043],{},"原理如下：",[3044,3045,3046,3059],"table",{},[3047,3048,3049],"thead",{},[3050,3051,3052,3056],"tr",{},[3053,3054,3055],"th",{},"属性",[3053,3057,3058],{},"说明",[3060,3061,3062,3080],"tbody",{},[3050,3063,3064,3069],{},[3065,3066,3067],"td",{},[920,3068,3026],{},[3065,3070,3071,3072,3075,3076,3079],{},"值为对象，包含所有被",[920,3073,3074],{},"ref","属性标识的",[920,3077,3078],{},"DOM","元素或组件实例。",[3050,3081,3082,3086],{},[3065,3083,3084],{},[920,3085,3035],{},[3065,3087,3088],{},"值为对象，当前组件的父组件实例对象。",[989,3090,3092],{"id":3091},"_7-provideinject","7. provide、inject",[989,3094,3096],{"id":3095},"_8-pinia-方式","8. Pinia 方式",[999,3098],{"title":3099},"请参考 Pinia 详解章节",[989,3101,3103],{"id":3102},"_9-slot-插槽","9. Slot 插槽",[3105,3106,3107],"style",{},"html pre.shiki code .sxrX7, html code.shiki .sxrX7{--shiki-light:#24292E;--shiki-default:#24292E;--shiki-dark:#E1E4E8}html pre.shiki code .sovSZ, html code.shiki .sovSZ{--shiki-light:#22863A;--shiki-default:#22863A;--shiki-dark:#85E89D}html pre.shiki code .snPdu, html code.shiki .snPdu{--shiki-light:#6F42C1;--shiki-default:#6F42C1;--shiki-dark:#B392F0}html pre.shiki code .sIIMD, html code.shiki .sIIMD{--shiki-light:#032F62;--shiki-default:#032F62;--shiki-dark:#9ECBFF}html .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 .sCsY4, html code.shiki .sCsY4{--shiki-light:#6A737D;--shiki-default:#6A737D;--shiki-dark:#6A737D}html pre.shiki code .sBjJW, html code.shiki .sBjJW{--shiki-light:#005CC5;--shiki-default:#005CC5;--shiki-dark:#79B8FF}html pre.shiki code .sP4rz, html code.shiki .sP4rz{--shiki-light:#E36209;--shiki-default:#E36209;--shiki-dark:#FFAB70}",{"title":11,"searchDepth":1132,"depth":1132,"links":3109},[3110,3111,3112,3113,3114,3115,3117,3118,3119],{"id":991,"depth":1132,"text":994},{"id":1606,"depth":1132,"text":1609},{"id":1800,"depth":1132,"text":1803},{"id":2228,"depth":1132,"text":2231},{"id":2675,"depth":1132,"text":2678},{"id":3010,"depth":1132,"text":3116},"6. $refs、$parent",{"id":3091,"depth":1132,"text":3092},{"id":3095,"depth":1132,"text":3096},{"id":3102,"depth":1132,"text":3103},"md",{},{"title":816,"description":817},"tutorials\u002Ffront-end\u002Fvue3\u002Fcomponent-communication","FQiEIG9pZZgec-OeaaNZJRe7AixH4dWvoR-e_c_Rqn0",1775496421093]