当前位置: 首页 > news >正文

从一段模板说起

<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({...})
http://www.jsqmd.com/news/1089345/

相关文章:

  • 视觉问答(VQA)的技术演进、核心挑战与未来展望
  • SQLite 在独立开发中的实战与优化:用轻量架构应对高并发
  • PilotGo-plugin-llmops vs 传统运维工具:为什么AI驱动是未来趋势
  • Web应用文件上传漏洞实战:从SPON系统漏洞看安全防御
  • Android NFC 实战:从权限配置到地铁卡数据解析
  • 从SHP到Excel,再到CAD:一站式地理数据格式转换与互操作实战指南
  • [智能体-589]:OpenClaw:HTML、JavaScript 、TypeScript、 Node.js、Python在智能体技术栈中各自的作用对比
  • Proteus原理图高效布线:标签与总线的进阶应用指南
  • OpenCore Legacy Patcher:三步让老旧Mac重获新生,体验最新macOS系统
  • 硬件原理图设计审查实战指南:从Checklist到高效协作
  • Linux内核页缓存覆写提权双链攻击深度剖析:CVE-2026-46331与CVE-2026-43503联动利用、检测与加固实战
  • 《淘宝订单API为什么个人账号调不通?企业认证+场景核验避坑指南》(附python源码)
  • 深入解析ChatGPT API的Token机制:从原理到精准计费实践
  • 【Linux】ClamAV实战:从零构建自动化病毒扫描与邮件告警系统
  • BetterGI 0.38.1版本安装失败终极解决方案:三步快速修复指南
  • QMCDecode:一键解锁QQ音乐加密格式,让音乐回归自由
  • 基于MCP协议与真实浏览器的AI驱动自动化测试实践
  • 原神帧率解锁技术方案:基于内存写入的安全高帧率实现
  • 从零到一:在VS2022中集成QT的实战环境配置
  • 如何用Python打造智能抢票神器:大麦网自动抢票脚本终极指南
  • CodeWarrior 调试实战:从断点到变量窗格的排错指南
  • 终极指南:如何用OneMore插件轻松实现OneNote全局搜索替换,告别手动修改烦恼!
  • 如何通过OneMore插件高效管理OneNote笔记:从基础编辑到智能组织实践指南
  • 【PyTorch】从ModuleNotFoundError到模型洞察:torchinfo安装、实战与避坑指南
  • 从手动到脚本:探索文件资源管理器(explorer)的优雅重启与状态恢复
  • EhViewer开源漫画应用:从零开始打造个性化漫画阅读体验的完整指南
  • 告别繁琐配置:基于Env与CLion的RT-Thread现代化开发环境一站式搭建
  • 抖音无水印下载终极指南:5分钟学会批量保存高清视频
  • Windows Cleaner:告别C盘爆红,让你的电脑重获新生
  • AMD Ryzen调试工具终极掌控:深度挖掘SMUDebugTool完全解锁指南