Vue.js 的运作流程大致可以分为以下几个核心阶段:
-
初始化:
-
创建 Vue 实例:使用
new Vue({ ... })
创建一个新的 Vue 实例,传入一个配置对象,其中包含了数据(data
)、计算属性(computed
)、方法(methods
)、生命周期钩子(created
、mounted
等)、模板/渲染函数(template
或render
)以及其它选项(如props
、watch
、mixins
、directives
等)。 -
选项合并与数据观测:Vue.js 会合并 Vue 实例及其父级组件(如果有)的所有选项,并对数据对象进行响应式处理。Vue 使用 Object.defineProperty 方法为数据对象的每个属性定义 getter 和 setter,从而实现数据变化的追踪。
-
-
模板编译(仅适用于使用模板的情况):
- 解析模板:Vue 解析模板字符串,将其转换为抽象语法树(AST)。
- 优化与静态节点标记:Vue 对 AST 进行静态节点分析,标记静态内容,以提升更新效率。
- 生成渲染函数:基于解析和优化后的 AST,Vue 生成一个 JavaScript 函数(即渲染函数),该函数负责将数据对象映射为最终的 DOM 结构。
-
挂载:
- 创建虚拟 DOM:Vue 调用渲染函数,将数据对象转化为虚拟 DOM 树(VNode)。
- DOM 渲染:Vue 使用 Vue 自己的 diff 算法 将虚拟 DOM 树与现有 DOM 进行对比,最小化地更新实际 DOM,以反映数据的变化。
- 生命周期钩子:在挂载过程中,Vue 会依次触发
beforeCreate
、created
、beforeMount
、mounted
等生命周期钩子函数,允许开发者在特定阶段执行自定义逻辑。
-
数据驱动的视图更新:
- 数据变更:当 Vue 实例的数据发生变化时,响应式系统会触发相应的 setter,记录变化。
- 依赖通知与视图更新:setter 通知依赖它的 watcher(即观察者),这些 watcher 可能是计算属性、组件、指令等。Watcher 根据数据变化触发更新,重新执行渲染函数生成新的 VNode。
- 差异更新:Vue 使用 diff 算法比较新旧 VNode,找出最小化的 DOM 更新操作序列,高效地更新实际 DOM。
- 生命周期钩子:在视图更新过程中,可能触发
beforeUpdate
和updated
生命周期钩子。
-
组件化:
- 组件注册与复用:开发者可以定义和注册多个组件,每个组件有自己的模板、数据、方法等。组件可以嵌套使用,形成复杂的视图结构。
- 组件间通信:通过 props 向子组件传递数据,使用事件和
$emit
向父组件传递消息,或者借助 Vuex 等状态管理库实现跨层级通信。
-
路由管理(使用 Vue Router 时):
- 路由配置与导航:设置路由规则,定义路由组件。用户通过点击链接、使用编程方式或浏览器地址栏直接输入 URL 导航到不同路由。
- 路由守卫:全局和组件级别的路由守卫(如
beforeEach
、beforeRouteEnter
等)在路由切换前后执行,可以用来控制路由访问权限、数据预取、页面过渡效果等。 - 动态组件加载:Vue Router 支持懒加载路由组件,仅在路由被访问时才加载对应的组件代码。
-
销毁:
- 卸载组件:当组件不再需要时(如路由切换离开组件、父组件销毁等),Vue 会触发
beforeDestroy
和destroyed
生命周期钩子,释放组件持有的资源。 - 解除数据观测:Vue 会清理响应式数据观测,撤销对数据对象属性的 getter/setter 定义,停止对数据变化的追踪。
- 卸载组件:当组件不再需要时(如路由切换离开组件、父组件销毁等),Vue 会触发
以上就是 Vue.js 的基本运作流程,涵盖了从实例创建到组件销毁的全周期。Vue 以其高效的响应式系统、组件化架构和灵活的路由管理,为开发者提供了构建复杂单页应用的强大工具。