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

相关文章

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

Qt QCustomPlot库简介(最新推荐)

《QtQCustomPlot库简介(最新推荐)》QCustomPlot是一款基于Qt的高性能C++绘图库,专为二维数据可视化设计,它具有轻量级、实时处理百万级数据和多图层支持等特点,适用于科学计算、... 目录核心特性概览核心组件解析1.绘图核心 (QCustomPlot类)2.数据容器 (QCPDataC

MySQL中的表连接原理分析

《MySQL中的表连接原理分析》:本文主要介绍MySQL中的表连接原理分析,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、背景2、环境3、表连接原理【1】驱动表和被驱动表【2】内连接【3】外连接【4编程】嵌套循环连接【5】join buffer4、总结1、背景

python中Hash使用场景分析

《python中Hash使用场景分析》Python的hash()函数用于获取对象哈希值,常用于字典和集合,不可变类型可哈希,可变类型不可,常见算法包括除法、乘法、平方取中和随机数哈希,各有优缺点,需根... 目录python中的 Hash除法哈希算法乘法哈希算法平方取中法随机数哈希算法小结在Python中,

Java Stream的distinct去重原理分析

《JavaStream的distinct去重原理分析》Javastream中的distinct方法用于去除流中的重复元素,它返回一个包含过滤后唯一元素的新流,该方法会根据元素的hashcode和eq... 目录一、distinct 的基础用法与核心特性二、distinct 的底层实现原理1. 顺序流中的去重

前端如何通过nginx访问本地端口

《前端如何通过nginx访问本地端口》:本文主要介绍前端如何通过nginx访问本地端口的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、nginx安装1、下载(1)下载地址(2)系统选择(3)版本选择2、安装部署(1)解压(2)配置文件修改(3)启动(4)

HTML中meta标签的常见使用案例(示例详解)

《HTML中meta标签的常见使用案例(示例详解)》HTMLmeta标签用于提供文档元数据,涵盖字符编码、SEO优化、社交媒体集成、移动设备适配、浏览器控制及安全隐私设置,优化页面显示与搜索引擎索引... 目录html中meta标签的常见使用案例一、基础功能二、搜索引擎优化(seo)三、社交媒体集成四、移动

HTML input 标签示例详解

《HTMLinput标签示例详解》input标签主要用于接收用户的输入,随type属性值的不同,变换其具体功能,本文通过实例图文并茂的形式给大家介绍HTMLinput标签,感兴趣的朋友一... 目录通用属性输入框单行文本输入框 text密码输入框 password数字输入框 number电子邮件输入编程框

HTML img标签和超链接标签详细介绍

《HTMLimg标签和超链接标签详细介绍》:本文主要介绍了HTML中img标签的使用,包括src属性(指定图片路径)、相对/绝对路径区别、alt替代文本、title提示、宽高控制及边框设置等,详细内容请阅读本文,希望能对你有所帮助... 目录img 标签src 属性alt 属性title 属性width/h

CSS3打造的现代交互式登录界面详细实现过程

《CSS3打造的现代交互式登录界面详细实现过程》本文介绍CSS3和jQuery在登录界面设计中的应用,涵盖动画、选择器、自定义字体及盒模型技术,提升界面美观与交互性,同时优化性能和可访问性,感兴趣的朋... 目录1. css3用户登录界面设计概述1.1 用户界面设计的重要性1.2 CSS3的新特性与优势1.