惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

S
Schneier on Security
Blog — PlanetScale
Blog — PlanetScale
L
LangChain Blog
P
Proofpoint News Feed
MongoDB | Blog
MongoDB | Blog
G
GRAHAM CLULEY
Simon Willison's Weblog
Simon Willison's Weblog
The Hacker News
The Hacker News
博客园_首页
W
WeLiveSecurity
Recorded Future
Recorded Future
S
Secure Thoughts
C
Check Point Blog
Y
Y Combinator Blog
Project Zero
Project Zero
量子位
www.infosecurity-magazine.com
www.infosecurity-magazine.com
S
Security Archives - TechRepublic
K
KPMG report finds enterprise disconnect between AI and its ROI | CIO
Scott Helme
Scott Helme
Spread Privacy
Spread Privacy
V
Vulnerabilities – Threatpost
AWS News Blog
AWS News Blog
S
Security @ Cisco Blogs
T
Threatpost
F
Full Disclosure
P
Proofpoint News Feed
T
The Exploit Database - CXSecurity.com
阮一峰的网络日志
阮一峰的网络日志
TaoSecurity Blog
TaoSecurity Blog
Last Week in AI
Last Week in AI
E
Exploit-DB.com RSS Feed
Microsoft Security Blog
Microsoft Security Blog
N
News | PayPal Newsroom
C
Cybersecurity and Infrastructure Security Agency CISA
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
C
Cisco Blogs
月光博客
月光博客
S
SegmentFault 最新的问题
B
Blog
GbyAI
GbyAI
J
Java Code Geeks
小众软件
小众软件
D
Docker
IT之家
IT之家
SecWiki News
SecWiki News
F
Fortinet All Blogs
Recent Commits to openclaw:main
Recent Commits to openclaw:main
Google Online Security Blog
Google Online Security Blog
NISL@THU
NISL@THU

博客园_首页

Plist 二进制格式 Milvus 和 PGVector,哪个更好? OpenClaw 已过时?在 VS Code 中运行 Hermes Agent! 第30篇文章:一个大三计科生的自白 Manim如何在数学公式中完美显示中文? Docker 部署 RocketMQ 5 并发编程核心概念辨析 C#事务处理最佳实践:别再让“主表存了、明细丢了”的破事发生 CLI 是什么?为什么大厂突然集体卷命令行? 【从0到1构建一个ClaudeAgent】协作-自主Agent UIImageView 设置图片不生效的原因排查 最小二乘问题详解20:无先验约束下的增量式SFM自由网平差 痞子衡嵌入式:大话双核i.MXRT1180之XIP应用里借助MU实现可靠Flash IAP的方法 AI Chat 封装, SemanticKerne.AiProvider.Unified 已发布 Windows下右键编辑js文件无法打开记事本——在注册表中使用环境变量 在后台服务中使用 Scoped 服务,为什么总是报错? H200 安装驱动并使用sglang启动模型 wireshark 抓包Trap上报告警内容 我用 AI 辅助开发了一系列小工具(2):图片压缩工具 [A Primer On MC and CC] 2.1 Memory Consistency 1 - 指令重排序和 SC 模型 Oracle数据库SCN推进技术详解与实践指南 玩转控件:封装个带图片的Label控件 Claude Code 4.7 真正该升级的不是模型,而是你的工作流 前端小白一句话,AI 帮我做了个颜值拉满的桌面媒体播放器。当代码不再是门槛,一句话编程就是现实。 5. WorkBuddy: 小龙虾的灵魂三件套,让你的小龙虾不只是工具 SQLite 分片方案实战:三种分片策略的深度对比 告别简陋 UI!一款基于 Fluent Design 和基于 WinUI 的开源免费、现代化的 Avalonia UI 控件库 关于二进制排列组合枚举的总结 AI开发-python-LangGraph框架(3-27-LangGraph从零实现大模型智能决策工作流) ElasticSearch主分片和副本分片概念详解 【002】HTTPS 粗解:证书、TLS 握手与对后端配置的影响 Hermes Agent 一周暴涨五万 Star,但我劝你别急着追 明明连接的是Redis的DB0,为什么能查到DB3的数据? 【从0到1构建一个ClaudeAgent】协作-Agent团队 熟悉电子元器件之后,电子小白下一步该怎么走? MAF快速入门(23)通过C#类定义Skills .NET 高级开发 | 手写一个对象映射框架 FastAPI数据库ORM怎么选?我肝了三个Demo后,终于不再纠结了 mysqldump 参数拾遗:在遗忘与铭记之间 C# .NET 周刊|2026年3月5期 Claude code入门 - 陈彦斌 一文学习入门 ThingsBoard 开源物联网平台 GitHub 热门项目 | 2026年04月16日 如何为GIT设置全局勾子,为每次提交追加信息 Number.isFinite和isFinite与isNaN()和Number.isNaN的区别 PortSwigger SQL注入LAB2 推荐一个测试人必备的Skills,从功能到性能全搞定(附详细实操和安装下载方式) 筑基期:掌握Odoo基础核心知识点02(Odoo XML 开发方式详解) GLM模型这么火,咱们用vllm也咧一个呗! 深入理解 AbortController:从底层原理到跨语言设计哲学 字符串学习笔记 多租户系统框架的基础模块设计和分析设计 Apache SeaTunnel Zeta 为什么能做到“又快又稳”? AI开发-python-LangGraph框架(3-26-LangGraph基本概念及第一个简单样例) Vue 3 组件通信,别只会用 Props 和 Emits 了,这几个狠活儿你得看看 ElasticSearch7.X版本配置密码 用Manim实现动态交点计算--从一个动点问题说起 团结引擎+Addressable+Instant Game打包抖音小游戏 function call 实战:让 LLM 自动判断 pod 异常、调用日志工具并完成故障分析 bubseek —— 让 Agent 的足迹,变成团队的洞察 通过 C# 读取并导出 PDF 书签 如何用 GitHub Actions 实现 Steam 自动化发布 【从0到1构建一个ClaudeAgent】并发-后台任务 .NET 高级开发 | 定制 ASP.NET Core 框架 电子小白:什么是运算放大器(运放) zero2Agent:面向大厂面试的 Agent 工程教程,从概念到生产的完整学习路线 堆上的ORW HC32F460 USB CDC通信异常:非对齐访问异常排查 20260413-Hyperbridge 攻击事件:发生在默克尔山上的验证绕过 那些喊着AI 要淘汰你的人,正在靠你的焦虑赚大钱! 深度学习进阶(八)Swin Transformer 最小二乘问题详解19:带先验约束的增量式SFM优化与实现 SnapTranslate 3.0 正式发布:全局划词翻译 + 完整英语学习闭环,一站式搞定查词、记词、复习 工作的意义、工作的困难认知再思考 .NET + AI 进阶实战:基于类的技能开发 - 打造可治理的 Agent 能力模块 【从0到1构建一个ClaudeAgent】规划与协调-技能 上周热点回顾(4.6-4.12) 电子小白的工具三件套:面包板、杜邦线、万能板 单表五亿数据的查询优化 | Mysql、StarRocks 2. WorkBuddy:从“我是谁”到“帮我干活” C# 如何减少代码运行时间:7 个实战技巧 基于HelixToolkit.SharpDX 渲染3D模型 - 笺上知微 从零开始的双臂具身VLA起源及现阶段发展综述 - SkyXZ 记对 xonsh shell 的使用, 脚本编写, 迁移及调优 - pluvium27 受够了Vibe Coding的失控?换个起点,让AI事半功倍 从开始配置漏洞环境到漏洞复现流程 - 難しい 关于10年工作经验的程序员对OpenClaw的实战经验分享以及看法 - 虚无境 Any metadata 的内存布局 C# .NET 周刊|2026年3月2期 - InCerry 我帮你测过了,测试圈排名第二的 Skill 依然很牛逼 Skill Discovery | 无监督技能发现的经典工作总结 - MoonOut 上下文工程是什么?过时了么?一文讲明白! - 一枫说码 开了 TUN 模式还是直连?90% 的人都踩过这个坑 AScript扩展多种脚本语言 - rockey627 AI 学习笔记:Agent 的记忆机制 你能被装进一个文件里吗?——7 万人把同事"蒸馏"成了 AI - 我没有三颗心脏 Claude Code 通关手册(七):给 AI 装上技能包——Skills 完全指南 - 暮色之狐 在浏览器中快速编辑代码:VSCode Web 集成实践 - Newbe36524 蒸馏自己 skill?基于 Deepseek 的蒸馏器,丐版蒸馏方式,简单便捷 - To_Carpe_Diem Spring AI Aliababa和AgentScope,哪个更好? - 苏三说技术
Vue 组件的本质
微山湖上静悄悄 · 2026-06-15 · via 博客园_首页

从一段模板说起

<div id="app">
  <my-header :title="pageTitle"></my-header>
  <my-content :posts="posts"></my-content>
  <my-footer></my-footer>
</div>

<my-header> 是什么?HTML 标准里没有这个标签。但 Vue 能理解它,把它变成一个完整的 DOM 子树。

这就是组件的本质:把一段模板、逻辑、样式打包成一个"自定义标签",像搭乐高一样搭页面。


组件是什么?

从数据结构的角度看,一个 Vue 组件就是一堆配置选项

{
  name: 'MyComponent',
  props: ['title'],
  data() { return { count: 0 } },
  computed: { ... },
  methods: { ... },
  template: '<div>{{ title }} - {{ count }}</div>'
}

当 Vue 遇到 <my-component>,它做的事情是:

  1. 拿出这个配置对象
  2. 创建一个组件实例
  3. 把实例挂载到 DOM 树中

组件的注册

全局注册

Vue.component('my-component', {
  template: '<div>我是一个全局组件</div>'
})

内部实现(简化):

Vue.component = function(id, definition) {
  // 把组件配置存到全局的 options.components 中
  this.options.components[id] = definition
}

全局组件在任何模板中都可以使用,因为 Vue 会在找不到某个标签对应的原生 HTML 元素时,去 Vue.options.components 中查找。

局部注册

new Vue({
  components: {
    'my-component': { template: '<div>局部组件</div>' }
  }
})

局部组件只在当前实例(及其子组件)的模板中可用。


组件实例的创建过程

当 Vue 在模板中遇到一个组件标签时,发生了什么?

模板: <my-component :title="hello"></my-component>

第1步:识别这是一个组件(不是原生 HTML 标签)
第2步:找到组件的配置对象(从全局/局部注册中)
第3步:创建组件实例 new VueComponent(options)
第4步:执行组件的生命周期(beforeCreate → created → beforeMount → mounted)
第5步:把组件的渲染结果(VNode)插入父组件的 VNode 树中

创建组件实例的核心代码(简化):

function createComponentInstance(vnode, parent) {
  const options = {
    _isComponent: true,
    _parentVnode: vnode,     // 组件在父模板中对应的 VNode
    parent: parent            // 父组件实例
  }

  // 调用 VueComponent 构造函数
  return new vnode.componentOptions.Ctor(options)
}

组件通信

父→子:Props

父组件传值给子组件:

<child :name="parentName"></child>

Props 的工作原理:

  1. 父组件的渲染函数中,parentName 是父组件 data 的一个响应式属性
  2. 子组件的 props 声明了 name
  3. Vue 在创建子组件时,把父组件传递的值赋给子组件的 _props.name
  4. 子组件的 _props 也是响应式的——所以父组件的数据变了,子组件也会自动更新

简化实现:

function initProps(vm, propsOptions, propsData) {
  vm._props = {}

  propsOptions.forEach(key => {
    // 为每个 prop 创建响应式属性
    defineReactive(vm._props, key, propsData[key])

    // 代理到 vm 上,让子组件可以用 this.name 访问
    Object.defineProperty(vm, key, {
      get() { return vm._props[key] },
      set() { console.warn('不要直接修改 prop!') }
    })
  })
}

子→父:$emit

// 子组件
this.$emit('add', payload)

// 父模板
<child @add="handleAdd"></child>

$emit 的原理非常直白:

Vue.prototype.$emit = function(event, ...args) {
  // 找到父组件在当前组件的 VNode 上绑定的事件监听器
  const listeners = this._parentVnode.componentOptions.listeners
  if (listeners && listeners[event]) {
    listeners[event](...args)  // 直接调用
  }
}

就是把父组件传进来的回调函数存起来,$emit 的时候调用。

跨层级:Provide / Inject

// 祖先
provide() {
  return { theme: 'dark' }
}

// 后代(不管隔了多少层)
inject: ['theme']

实现原理简化:

function initProvide(vm) {
  // 把 provide 的结果合并到父组件的 _provided 上
  vm._provided = Object.create(vm.$parent ? vm.$parent._provided : null)
  const provide = vm.$options.provide
  if (typeof provide === 'function') {
    Object.assign(vm._provided, provide.call(vm))
  }
}

function initInjections(vm) {
  // 沿原型链向上查找
  const result = resolveInject(vm.$options.inject, vm)
  Object.keys(result).forEach(key => {
    defineReactive(vm, key, result[key])
  })
}

function resolveInject(inject, vm) {
  const result = {}
  for (const key of inject) {
    let source = vm
    while (source) {
      if (source._provided && key in source._provided) {
        result[key] = source._provided[key]
        break
      }
      source = source.$parent  // 向上查找
    }
  }
  return result
}

利用了 JS 的原型链:vm._provided 的原型指向父组件的 _provided,向上查找自然遍历了整个组件树。


插槽(Slot)

插槽的本质是"父组件写内容,子组件用占位符接收"

<!-- 使用组件 -->
<my-layout>
  <h1 slot="header">我的标题</h1>   ← 父组件定义内容
  <p>正文内容</p>
</my-layout>

<!-- 组件定义 -->
<div class="layout">
  <slot name="header"></slot>        ← 子组件占位
  <slot></slot>                      ← 默认插槽
</div>

插槽的 VNode 是在父组件的上下文中渲染的,所以插槽内容访问的是父组件的数据。但插槽的挂载位置是在子组件内部。


一个完整的组件生命周期

new Vue({...})
  │
  ▼
beforeCreate  ← 数据/方法还没初始化,不能访问 data
  │
  ▼
created      ← 数据/方法/计算属性都已初始化,可以访问
  │
  ▼
beforeMount  ← 模板编译完成,但还没挂载到 DOM
  │
  ▼
mounted      ← 挂载完成,可以访问 DOM
  │
  ▼
(数据变化)
  │
  ▼
beforeUpdate ← 数据变了,但 DOM 还没更新
  │
  ▼
updated      ← DOM 已更新
  │
  ▼
beforeDestroy ← 销毁之前,清理定时器/事件监听的好时机
  │
  ▼
destroyed     ← 组件已销毁

总结

概念 本质
组件注册 把配置对象存到字典里,模板解析时查找
组件实例 根据配置对象 + 父组件上下文 new 出的 VueComponent
Props 父→子,响应式传递
$emit 调用父组件绑定的回调函数
Provide/Inject 原型链向上查找
Slot 父作用域渲染,子组件占位挂载

组件系统让 Vue 应用变成了一棵可组合的实例树。每个组件都是一个独立的"小 Vue",有自己的响应式数据、有自己的 Watcher、有自己的生命周期。

但组件模板中写的 <div>{{ message }}</div>,Vue 是怎么把它变成能执行的 JavaScript 函数的?