【vue2源码】模版编译

文章目录

  • 一、mount 基本流程
  • 二、执行 $mount 方法
  • 三、模版编译
    • 1、入口代码
    • 2、parse
      • 2.1 parseHTML
      • 2.2 parseText
    • 3、generate
      • genElement 函数
    • 4、createCompileToFunctionFn
  • 4、mountComponent

一、mount 基本流程

在执行 _init (new Vue时) 的方法中,调用了 vm.$mount(vm.$options.el) 后的挂载流程:

  1. 通过 parse 将模版编译成抽象语法树 ast
  2. 将 ast 转成 render 函数
  3. 执行 render 生成 vnode
  4. 通过 mountComponent,执行 patch 将 vnode 变成真实 dom

二、执行 $mount 方法

源码位置: src/platforms/web/runtime-with-compiler.ts

// 保留了 Vue 原型上原始的 $mount 方法的引用
const mount = Vue.prototype.$mount// 定义了一个新的 $mount 方法
Vue.prototype.$mount = function (el?: string | Element,hydrating?: boolean
): Component {el = el && query(el)const options = this.$options// resolve template/el and convert to render functionif (!options.render) {let template = options.templateif (template) {// ...} else if (el) {// @ts-expect-errortemplate = getOuterHTML(el)}if (template) {// compileToFunctions 方法会将 template 编译成 render 函数const { render, staticRenderFns } = compileToFunctions(template,{// ...},this)options.render = render}}// 调用原始 $mount return mount.call(this, el, hydrating)
}

三、模版编译

1、入口代码

源码路径: src/compiler/index.ts

export const createCompiler = createCompilerCreator(function baseCompile(template: string,options: CompilerOptions
): CompiledResult {// 解析模板字符串生成 ASTconst ast = parse(template.trim(), options)// 对AST进行优化// ...// 使用 generate 函数将 AST 转换为渲染函数的代码字符串。// 这一步是将结构化的 AST 转换为实际可执行的 JavaScript 代码const code = generate(ast, options)return {ast,render: code.render,staticRenderFns: code.staticRenderFns}
})

2、parse

parse函数的作用:用于将模板字符串转换为抽象语法树(AST)
源码路径: src/compiler/parser/index.ts
基本结构:

export function parse(template: string, options: ComponentOptions) {// console.log("模版解析 parse");const stack: any[] = [];let root; // 最终生成的 AST  let currentParent;parseHTML(template, {start(tag, attrs) {// 当遇到标签起始处的处理,创建 AST 元素节点let element: ASTElement = createASTElement(tag, attrs, currentParent);if (!root) {root = element;}currentParent = element;processRawAttrs(element);// 进栈stack.push(element);},// 匹配到结束标签后的处理end() {// 当遇到标签结束处的处理// 弹出栈,更新当前处理的父级节点const element = stack[stack.length - 1];stack.length -= 1;currentParent = stack[stack.length - 1];if (currentParent) {currentParent.children.push(element);}},chars(text: string) {// 文本内容处理const children = currentParent.children;text = text.trim();if (text) {let child: ASTNode;let res;// parseText 的实现在下面)(2.2)if (text !== " " && (res = parseText(text))) {// 解析文本,这里是带有 {{}} 的情况// console.log(res);child = {type: 2,expression: res.expression,tokens: res.tokens,text,};} else {// 文本节点child = {type: 3,text,};}if (child) {children.push(child);}}},comment(text: string) {// 注释的处理}});return root;
}

2.1 parseHTML

parseHTML的工作原理基于正则表达式,逐步读取HTML字符串,并且根据标签的开始、结束、文本内容等来构建AST

源码路径:src/compiler/parser/html-parser.ts

下面是我自己手写的 parseHTML ,不考虑注释,自闭合标签等。

import { ASTAttr } from "src/types/compiler";interface HTMLParserOptions {start?: Function;end?: Function;chars?: Function;comment?: (content: string) => void;
}const attribute =/^\s*([^\s"'<>\/=]+)(?:\s*(=)\s*(?:"([^"]*)"+|'([^']*)'+|([^\s"'=<>`]+)))?/;
const startTagOpen = /^<([a-zA-Z_]+[0-9]*)/;
const startTagClose = /^\s*(\/?)>/;
const endTag = /^<\/([a-zA-Z_]+[0-9]*)>/;export function parseHTML(html: string, options: HTMLParserOptions) {const stack: any[] = [];let index = 0; // 指针let last; // 剩余部分while (html) {last = html;let textEnd = html.indexOf("<");if (textEnd == 0) {// Comment:// ... // Doctype:// ...// 结束标签const endTagMatch = html.match(endTag);if (endTagMatch) {advance(endTagMatch[0].length);parseEndTag(endTagMatch[1]);continue;}// 开始标签const startTagMatch = parseStartTag();if (startTagMatch) {handleStartTag(startTagMatch);continue;}}let rest, text;if (textEnd >= 0) {// 标签内有文本rest = html.slice(textEnd);text = html.substring(0, textEnd);}// 处理标签内的文本if (text) {advance(text.length);}// 调用文本的处理if (options.chars && text) {options.chars(text);}if (html === last) {index++;html = html.substring(1);}}function advance(n) {index += n;html = html.substring(n);}// 解析开始标签function parseStartTag() {const start = html.match(startTagOpen);if (start) {const match: any = {tagName: start[1],attrs: [],start: index,};advance(start[0].length);// 处理开始标签的属性let attr, end;while (!(end = html.match(startTagClose)) &&(attr = html.match(attribute))) {// 当不为 ">" 且匹配到属性时attr.start = index;advance(attr[0].length);attr.end = index;match.attrs.push(attr);}// 开始标签的 >if (end) {advance(end[0].length);match.end = index;return match;}}}// 处理开始标签function handleStartTag(match) {const tagName: string = match.tagName;// 处理属性const len = match.attrs.length;const attrs: ASTAttr[] = new Array(len);for (let i = 0; i < len; i++) {const args = match.attrs[i];const value = args[3] || args[4] || args[5] || "";attrs[i] = {name: args[1],value: value,};}// 开始标签进栈stack.push({tag: tagName,lowerCasedTag: tagName.toLocaleLowerCase(),attrs: attrs,start: match.start,end: match.end,});if (options.start) {options.start(tagName, attrs, match.start, match.end);}}// 解析结束标签function parseEndTag(tagName: string) {const lastStack = stack[stack.length - 1];if (tagName && tagName.toLocaleLowerCase() === lastStack.lowerCasedTag) {if (options.end) {options.end(lastStack.tag);}stack.length = stack.length - 1;}}
}

2.2 parseText

parseText 函数是模板编译过程的一部分,用于解析文本节点中的插值表达式

源码路径:src/compiler/parser/text-parser.ts

// 解析给定文本text中的动态绑定表达式,并返回一个包含解析结果的对象
export function parseText(text: string): TextParseResult | void {const tagRE = defaultTagRE;const tokens: string[] = [];const rawTokens: any[] = [];// 定义一个 lastIndex 变量,用于记录上一次匹配的位置let lastIndex = (tagRE.lastIndex = 0);let match, index, tokenValue;while ((match = tagRE.exec(text))) {// 这里是匹配 {{ }}index = match.index;// 文本(这里是 {{}} 前面的文本)if (index > lastIndex) {rawTokens.push((tokenValue = text.slice(lastIndex, index)));tokens.push(JSON.stringify(tokenValue));}debugger// {{}} 中的内容const exp = match[1].trim();tokens.push(`_s(${exp})`);rawTokens.push({ "@binding": exp });lastIndex = index + match[0].length;}// 判断 lastIndex 变量是否小于文本长度,小于则代表 {{}} 后面还有文本if (lastIndex < text.length) {rawTokens.push((tokenValue = text.slice(lastIndex)));tokens.push(JSON.stringify(tokenValue));}// return 生成示例:// 比如:<div>msg: {{ message }}</div>// 返回:// {//   expression: "\"msg:\"+_s(message)",//   tokens: [//     "msg:",//     {//         "@binding": "message"//     }//   ]// }return {expression: tokens.join("+"),tokens: rawTokens,};
}

3、generate

generate 函数的主要作用是基于给定的AST生成相应的JavaScript代码(渲染函数)
这个渲染函数将会返回一个虚拟节点(VNode)树,表示组件的DOM结构

源码路径:src/compiler/codegen/index.ts

export function generate (ast,options
) {const state = new CodegenState(options);const code = ast ? genElement(ast, state) : '_c(div)';return {render: `with(this){return ${code}}`,staticRenderFns: state.staticRenderFns}
}

code 生成的示例:

_c('div',{attrs:{"id":"app"}},[(show)?_c('h3',{staticClass:"active"},[_v("message: "+_s(message))]):_e()])

genElement 函数

主要职责是将抽象语法树(AST)的元素(Element)节点转换成字符串形式的渲染函数代码。该过程涉及到递归地处理元素的所有属性、指令和子节点,以确保能生成准确反映模板结构和逻辑的渲染函数代码

核心工作内容:
1、处理元素的属性和指令

genElement需要将元素上的所有属性(包括静态属性和动态绑定的属性)和指令(如v-if、v-for、v-model等)转换成 JavaScript 代码。对于指令,这通常意味着生成特定的代码来实现指令定义的行为。

2、处理子节点
对于每个元素节点,genElement还需要考虑其子节点。这包括:

  • 递归地对子元素调用genElement,生成子元素的渲染函数代码。
  • 将文本节点转换成_v(创建文本 VNode 的函数)调用。
  • 将表达式节点转换成_s(toString 包装器)调用,以确保任何绑定的表达式都可以正确地转换成字符串。

3、生成渲染函数代码
最终,genElement需要生成类似于_c(‘div’, {…}, […])这样的函数调用代码。_c是创建元素 VNode 的函数,第一个参数是标签名,第二个参数是一个包含该元素所有属性和指令的数据对象,第三个参数是该元素的子节点数组。

4、处理插槽和组件
genElement还需要特别处理插槽和组件。
对于插槽,它需要生成_t(渲染插槽的函数)调用,并为插槽内容生成适当的代码。
对于组件,它需要根据组件定义生成_c(或特定于组件的创建函数,如果设置了functional标志)调用,并处理传递给组件的任何属性或事件监听器。

4、createCompileToFunctionFn

主要是将模板字符串编译成渲染函数,并且缓存了这个过程的结果以提高性能。
返回的compileToFunctions函数的主要作用是将 Vue 模板字符串转换成最终的渲染函数

以下是简化的createCompileToFunctionFn函数的示意性解释:

function createCompileToFunctionFn(compile) {const cache = Object.create(null);return function compileToFunctions(template, options, vm) {// 使用 options 和模板生成一个缓存的 keyconst key = options ? (options.delimiters ? String(options.delimiters) + template : template) : template;// 检查缓存中是否已经存在编译后的结果if (cache[key]) {return cache[key];}// 调用编译函数,将模板编译成 AST、优化后的 AST 和字符串形式的渲染函数const compiled = compile(template, options);// 将字符串形式的渲染函数转换成 JavaScript 函数const res = {};// 生成最终的渲染函数res.render = new Function(compiled.render);// 处理静态渲染函数,只有当使用了 v-once 指令时,这部分才不为空const staticRenderFns = compiled.staticRenderFns.map(code => new Function(code));res.staticRenderFns = staticRenderFns;// 缓存结果并返回cache[key] = res;return res;};
}

4、mountComponent

挂载组件的核心函数,它负责将一个 Vue 组件实例挂载到 DOM 上,并启动响应式更新机制,以便组件的状态改变时能自动更新对应的 DOM 表现

源码位置:src/core/instance/lifecycle.ts

核心代码:

function mountComponent(vm, el) {// 设置 vm.$el 以引用真实 DOM 元素vm.$el = el;// 如果没有定义 render 函数,尝试编译模板生成一个if (!vm.$options.render) {compileToRenderFunction(vm);}// 调用 beforeMount 生命周期钩子callHook(vm, 'beforeMount');// 创建观察者:在数据变化时重新渲染组件const updateComponent = () => {vm._update(vm._render(), hydrating);};// 创建组件级观察者,传递 updateComponent 作为更新函数new Watcher(vm, updateComponent, noop, {before() {callHook(vm, 'beforeUpdate');}}, true /* 表示这是一个组件观察者 */);// 挂载完成,调用 mounted 生命周期钩子if (vm.$vnode == null) {vm._isMounted = true;callHook(vm, 'mounted');}return vm;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/750518.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

力扣热题100_矩阵_240_搜索二维矩阵 II

文章目录 题目链接解题思路解题代码 题目链接 240. 搜索二维矩阵 II 编写一个高效的算法来搜索 m x n 矩阵 matrix 中的一个目标值 target 。该矩阵具有以下特性&#xff1a; 每行的元素从左到右升序排列。 每列的元素从上到下升序排列。 示例 1&#xff1a; 输入&#xf…

pytorch DDP模式下, 获取数据的的preftech + stream

直接上代码 - DDP forward if self.device_ids:if len(self.device_ids) 1:inputs, kwargs self.to_kwargs(inputs, kwargs, self.device_ids[0])output self.module(*inputs[0], **kwargs[0])else:inputs, kwargs self.scatter(inputs, kwargs, self.device_ids)outputs …

GAMES104-现代游戏引擎 1

主要学习重点还是面向就业&#xff0c;重点复习八股和算法 每天早上八点到九点用来学习这个课程 持续更新中... 第一节 游戏引擎导论 第二节 引擎架构分层

OLLAMA:如何像云端一样运行本地大语言模型

简介&#xff1a;揭开 OLLAMA 本地大语言模型的神秘面纱 您是否曾发现自己被云端语言模型的网络所缠绕&#xff0c;渴望获得更本地化、更具成本效益的解决方案&#xff1f;那么&#xff0c;您的探索到此结束。欢迎来到 OLLAMA 的世界&#xff0c;这个平台将彻底改变我们与大型…

橡胶工厂5G智能制造数字孪生可视化平台,推进橡胶工业数字化转型

橡胶5G智能制造工厂数字孪生可视化平台&#xff0c;推进橡胶工业数字化转型。随着信息技术的迅猛发展和智能制造的不断推进&#xff0c;数字化转型已成为制造业转型升级的重要方向。橡胶工业作为传统制造业的重要领域&#xff0c;正面临着产业升级和转型的迫切需求。橡胶5G智能…

软考79-上午题-【面向对象技术3-设计模式】-结构型设计模式02

一、组合模式 1-1、意图 将对象组合成树型结构&#xff0c;以表示"部分-整体"的层次结构。Composite使得用户对单个对象和组 合对象的使用具有一致性。 示例&#xff1a;对象&#xff1a;文件、文件夹 1-2、结构 Component 为组合中的对象声明接口&#xff1b;在适…

决策树 | 分类树回归树:算法逻辑

目录 一. 决策树(Decision Tree)1. 决策树的构建1.1 信息熵(Entropy)1.1.1 信息量&信息熵 定义1.1.2 高信息熵&低信息熵 定义1.1.3 信息熵 公式 1.2 信息增益(Information Gain)1.2.1 信息增益的计算1.2.2 小节 2. 小节2.1 算法分类2.2 决策树算法分割选择2.3 决策树算…

C# 单例模式

单例模式介绍 单例模式只允许被其自身实例化一次&#xff0c;且向外部提供了一个访问该实例的接口。 通常来说&#xff0c;单例对象进行实例化时一般不带参数&#xff0c;因为如果不同的实例化请求传递的参数不同的话会导致问题的产生。 单例模式主要特点 全局唯一性&#xf…

提升物流效率,快递平台实战总结与分享

随着电商行业的蓬勃发展&#xff0c;物流配送服务变得愈发重要。快递平台作为连接电商企业和消费者的桥梁&#xff0c;扮演着至关重要的角色。本篇博客将分享快递平台实战经验&#xff0c;总结关键要点&#xff0c;帮助物流从业者提升物流效率、优化服务质量。 ### 快递平台实…

汽车网络基础知识 要点

在以太网开发中&#xff0c;常常会听到一些专业名词&#xff0c;例如PHY&#xff0c;MAC&#xff0c;MII&#xff0c;switch&#xff0c;下面是解释 PHY PHY 是物理接口收发器&#xff0c;它实现物理层。包括 MII/GMII (介质独立接口) 子层、PCS (物理编码子层) 、PMA (物理介…

SQLiteC/C++接口详细介绍之sqlite3类(十四)

返回目录&#xff1a;SQLite—免费开源数据库系列文章目录 上一篇&#xff1a;SQLiteC/C接口详细介绍之sqlite3类&#xff08;十三&#xff09; 下一篇&#xff1a;SQLiteC/C接口详细介绍之sqlite3类&#xff08;十五&#xff09; 43.sqlite3_preupdate_hook sqlite3_preup…

PHP获取并分析过期删除.cn域名3-5数字(无4非0开)/2-4字母(声母)/三杂等品类域名

用途:获取分析最近过期删除.cn域名中3-5数字(无4非0开)/2-4字母(声母)/三杂等品类域名 先创建rec文件夹,用于按天缓存域名列表(不自动创建哦) 推荐宝塔灯自带的定时任务访问本文件网址&#xff0c;每早5-6点执行 获取两天后删除的cn域名名单并提取几个有价值类目的域名 <…

什么可以让你快速穿越低谷?

1. 变环境 当我们对当前生活状态感到彻底失望&#xff0c;当我们看透了人世间的冷暖&#xff0c;见证了人性的撕裂与拉扯&#xff0c;就会意识到需要改变环境。这可能包括搬离现居之地&#xff0c;寻找新的斗争之地。就像寻找宝玉发光之地一样&#xff0c;我们需要找到能让自己…

基于springboot的高校化学试剂仓储管理系统

文章目录 项目介绍主要功能截图&#xff1a;部分代码展示设计总结项目获取方式 &#x1f345; 作者主页&#xff1a;超级无敌暴龙战士塔塔开 &#x1f345; 简介&#xff1a;Java领域优质创作者&#x1f3c6;、 简历模板、学习资料、面试题库【关注我&#xff0c;都给你】 &…

VMware NSX Advanced Load Balancer (NSX ALB) 22.1.6 - 多云负载均衡平台

VMware NSX Advanced Load Balancer (NSX ALB) 22.1.6 - 多云负载均衡平台 应用交付&#xff1a;多云负载均衡、Web 应用防火墙和容器 Ingress 服务 请访问原文链接&#xff1a;https://sysin.org/blog/vmware-nsx-alb-22/&#xff0c;查看最新版。原创作品&#xff0c;转载请…

194 基于matlab的日历GUI制作

基于matlab的日历GUI制作&#xff0c;可实时显示当前的日期和时间&#xff0c;精确到秒。非常漂亮&#xff0c;也很基础&#xff0c;学习GUI的不错程序&#xff0c;程序已调通&#xff0c;可直接运行。 194 matlab 日历制作 GUI可视化 - 小红书 (xiaohongshu.com)

sqlplus设置提示符

作为DBA&#xff0c;需要管理好多数据库&#xff0c;经常会有一台服务器安装多个oracle实例的情况&#xff0c;为避免误操作实例&#xff0c;我们需要在执行sqkplus前&#xff0c;先通过$ echo $ORACLE_SID或 SQL>select name from v$database查看当前实例&#xff0c;这样难…

前端小白的学习之路(CSS3 一)

提示&#xff1a;CSS3 是 Cascading Style Sheets&#xff08;层叠样式表&#xff09;的第三个主要版本&#xff0c;引入了许多新的特性和增强功能&#xff0c;用于设计和布局网页。本章记录CSS3新增选择器&#xff0c;盒子模型。 目录 一、C3新增选择器 1) 属性选择器 1.[c…

Python语言基础与应用-北京大学-陈斌-P40-39-基本扩展模块/上机练习:计时和文件处理-给算法计时-上机代码

Python语言基础与应用-北京大学-陈斌-P40-39-基本扩展模块/上机练习&#xff1a;计时和文件处理-给算法计时-上机代码 上机代码&#xff1a; # 基本扩展模块训练 给算法计时 def factorial(number): # 自定义一个计算阶乘的函数i 1result 1 # 变量 result 用来存储每个数的阶…

小结:Node.js Express VS Koa

Express 是一个基于 Node.js 平台的极简、灵活的 web 应用开发框架&#xff0c;主要基于 Connect 中间件&#xff0c;并且自身封装了路由、视图处理等功能&#xff0c;使用人数众多。 Koa 是 Express 原班人马基于 ES6 新特性重新开发的框架&#xff0c;主要基于 co 中间件&am…