VUE源码分析4 -- Mustache简介 和重写Scanner

2024-05-13 06:38

本文主要是介绍VUE源码分析4 -- Mustache简介 和重写Scanner,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

下面我们说Vue Dom实现,vue dom 兼容mustache项目。这个链接是mustache官方工程 mustache github地址

请看官方mustahce语法文档,变量前后添加两个大括号{{}},代表要侦测的数据。Vue官方也是兼容改特性。

下面我们要重零开始写Mustache,为什么要重零开始写mustache呢,因为我们追踪vue dom底层,mustache是跑不了的。

 

熟悉Mustache项目

首先我们先看看Mustache 案例,

<!DOCTYPE html>
<html><head><meta charset="utf-8"><title></title><script src="https://unpkg.com/mustache@latest"></script>//引用mustache项目</head><body><div id="app"></div><div id="app1"></div></body><script type="text/javascript">console.log("here");var templateStr = `<ul>{{#arr}}<li>{{name}}的基本信息<div><p>姓名:{{name}}</p><p>性别:{{sex}}</p><p>年龄:{{age}}</p></div></li>{{/arr}}</ul>`;var data = {arr: [{"name": "小明","age": 12,"sex": "男"}, {"name": "小明","age": 12,"sex": "男"}, {"name": "小明","age": 12,"sex": "男"}]};var domStr = Mustache.render(templateStr, data);console.log(domStr);var container = document.getElementById('app');container.innerHTML=domStr;var templateStr1 = `		<ul>{{#arr}}<li>{{.}}</li>{{/arr}}</ul>`;var data1 ={arr:['A','B','C']}var domstr1 = Mustache.render(templateStr1, data1);console.log(domstr1);container.innerHTML +=domstr1;</script>
</html>

以上代码执行结果是

从上面给可以知道,{{name}},{{sex}},{{age}}是变量。并且{{#arr}}...{{/arr}}是否跟vue v-for相似。

 

Scanner类

下面分析mustache scanner类,为什么要先说scanner类呢,因为mustache把 模板字符串转成token,将tokens借和数据,解析为dom字符串。

例如,以下模板字符串是

<div><ol>{{#students}}<li>学生{{name}}的爱好是<ol>{{#hobbies}}<li>{{.}}</li>{{/hobbies}}</ol></li>{{/students}}</ol>
</div>

mustache内部先转成

[ ["text", "<div><ol>"],
["#", "students", [ ["text", "<li>学生"],
["name", "name"],
["text", "的爱好是<ol>"],
["#", "hobbies", [ ["text", "<li>"],
["name", "."],
["text", "</li>"],
]],
["text", "</ol></li>"],
]],
["text", "</ol></div>"] ]

为了验证以上步骤,我们修改官方mustache源码,把254代码改为

修改前:

修改后

然后网页输出信息为,可以看出此时模板字符串输出信息为tokens信息。

上面mustache源码在github链接下,请自行下载查看代码下载

 

下面我们开始重写Scanner类,

先创建Scanner.js文件,然后创建Scanner类,然后创建scan()和scanUtil()函数,scan函数是用于跳过某个字符,在mustache源码是跳过 {{ 和 }} 字符串。

export default class Scanner{constructor(templateStr){}scan(tag){//跳过 {{ 和 }}}scanUtil(stopTag){ //把字符串转成tokens}
}

scanUtil()函数功能用于把模板字符串转成tokens。

scan()是用于跳过某些字符串。

接下来我们继续写这两个函数代码,先看所有源码

Scanner.js

export default class Scanner{constructor(templateStr) {console.log(templateStr);this.templateStr = templateStr;//将模板字符串写道实例身上this.pos = 0;//当前位置this.tail = templateStr;}scan(tag){if(this.tail.indexOf(tag)==0){//第一位就是0this.pos += tag.length;this.tail = this.templateStr.substring(this.pos);}}scanUtil(stopTag){// console.log("scanUtil",stopTag);const pos_backup = this.pos;//当尾巴开头不是stopTag时候,说明没有扫描到到stopTag//写eos很必要,因为防止找不到,最后也要停止下来while(this.tail.indexOf(stopTag)!=0 && !this.eos()){this.pos++;//把尾巴改为当前扫描剩余字符串this.tail = this.templateStr.substring(this.pos);// console.log(this.tail);}return this.templateStr.substring(pos_backup,this.pos);}eos(){return this.pos >= this.templateStr.length;}
}

上面这个代码没什么好说的,都是基本截取字符串算法,自行手敲一次理解就行。

index.js

import Scanner from './Scanner.js'
//提供window全局方法
window.TemplateEngine = {render(templateStr,data){var scanner = new Scanner(templateStr);var word ;while(scanner.pos!=scanner.templateStr.length)// while(counter --){word = scanner.scanUtil("{{");console.log("word:",word);scanner.scan("{{");// console.log("word:",word);word = scanner.scanUtil("}}");console.log("word:",word);scanner.scan("}}");// console.log("word:",word);}}
}

在上面代码注意,在这里创建一个全局的对象TemplateEngine。然后一直循环截取字符串。

 

index.js

<!DOCTYPE html>
<html><head><meta charset="utf-8"><title></title><script src="/xuni/bundle.js" type="text/javascript" charset="utf-8"></script></head><body>213123</body><script type="text/javascript">var Str = "<h1>我买了一个{{thing}},好{{mood}}</h1>;"var data = {}TemplateEngine.render(Str)</script>
</html>

上面代码是创建一个模板字符串Str,把Str放到TemplateEngine.render()函数转换成tokens,然后查看浏览器console输出。

以上是mustache scanner类源码,其实很简单,就是把模板字符串转为tokens一些算法。感兴趣的小伙伴可以自行尝试,重写Scanner类源码地址在这里下载 点我下载

这篇关于VUE源码分析4 -- Mustache简介 和重写Scanner的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/984958

相关文章

部署Vue项目到服务器后404错误的原因及解决方案

《部署Vue项目到服务器后404错误的原因及解决方案》文章介绍了Vue项目部署步骤以及404错误的解决方案,部署步骤包括构建项目、上传文件、配置Web服务器、重启Nginx和访问域名,404错误通常是... 目录一、vue项目部署步骤二、404错误原因及解决方案错误场景原因分析解决方案一、Vue项目部署步骤

Springboot中分析SQL性能的两种方式详解

《Springboot中分析SQL性能的两种方式详解》文章介绍了SQL性能分析的两种方式:MyBatis-Plus性能分析插件和p6spy框架,MyBatis-Plus插件配置简单,适用于开发和测试环... 目录SQL性能分析的两种方式:功能介绍实现方式:实现步骤:SQL性能分析的两种方式:功能介绍记录

前端原生js实现拖拽排课效果实例

《前端原生js实现拖拽排课效果实例》:本文主要介绍如何实现一个简单的课程表拖拽功能,通过HTML、CSS和JavaScript的配合,我们实现了课程项的拖拽、放置和显示功能,文中通过实例代码介绍的... 目录1. 效果展示2. 效果分析2.1 关键点2.2 实现方法3. 代码实现3.1 html部分3.2

最长公共子序列问题的深度分析与Java实现方式

《最长公共子序列问题的深度分析与Java实现方式》本文详细介绍了最长公共子序列(LCS)问题,包括其概念、暴力解法、动态规划解法,并提供了Java代码实现,暴力解法虽然简单,但在大数据处理中效率较低,... 目录最长公共子序列问题概述问题理解与示例分析暴力解法思路与示例代码动态规划解法DP 表的构建与意义动

CSS弹性布局常用设置方式

《CSS弹性布局常用设置方式》文章总结了CSS布局与样式的常用属性和技巧,包括视口单位、弹性盒子布局、浮动元素、背景和边框样式、文本和阴影效果、溢出隐藏、定位以及背景渐变等,通过这些技巧,可以实现复杂... 一、单位元素vm 1vm 为视口的1%vh 视口高的1%vmin 参照长边vmax 参照长边re

CSS3中使用flex和grid实现等高元素布局的示例代码

《CSS3中使用flex和grid实现等高元素布局的示例代码》:本文主要介绍了使用CSS3中的Flexbox和Grid布局实现等高元素布局的方法,通过简单的两列实现、每行放置3列以及全部代码的展示,展示了这两种布局方式的实现细节和效果,详细内容请阅读本文,希望能对你有所帮助... 过往的实现方法是使用浮动加

css渐变色背景|<gradient示例详解

《css渐变色背景|<gradient示例详解》CSS渐变是一种从一种颜色平滑过渡到另一种颜色的效果,可以作为元素的背景,它包括线性渐变、径向渐变和锥形渐变,本文介绍css渐变色背景|<gradien... 使用渐变色作为背景可以直接将渐China编程变色用作元素的背景,可以看做是一种特殊的背景图片。(是作为背

C#使用DeepSeek API实现自然语言处理,文本分类和情感分析

《C#使用DeepSeekAPI实现自然语言处理,文本分类和情感分析》在C#中使用DeepSeekAPI可以实现多种功能,例如自然语言处理、文本分类、情感分析等,本文主要为大家介绍了具体实现步骤,... 目录准备工作文本生成文本分类问答系统代码生成翻译功能文本摘要文本校对图像描述生成总结在C#中使用Deep

CSS自定义浏览器滚动条样式完整代码

《CSS自定义浏览器滚动条样式完整代码》:本文主要介绍了如何使用CSS自定义浏览器滚动条的样式,包括隐藏滚动条的角落、设置滚动条的基本样式、轨道样式和滑块样式,并提供了完整的CSS代码示例,通过这些技巧,你可以为你的网站添加个性化的滚动条样式,从而提升用户体验,详细内容请阅读本文,希望能对你有所帮助...

css实现图片旋转功能

《css实现图片旋转功能》:本文主要介绍了四种CSS变换效果:图片旋转90度、水平翻转、垂直翻转,并附带了相应的代码示例,详细内容请阅读本文,希望能对你有所帮助... 一 css实现图片旋转90度.icon{ -moz-transform:rotate(-90deg); -webkit-transfo