深入理解 JavaScript 模板引擎
.markdown-body { line-height: 1.75; font-weight: 400; font-size: 15px; overflow-x: hidden; color: rgba(43, 43, 43, 1); font-family: -apple-system, system-ui, BlinkMacSystemFont, Helvetica Neue, PingFang SC, Hiragino Sans GB, Microsoft YaHei, Arial, sans-serif; background-image: linear-gradient(90deg, rgba(159, 219, 252, 0.15) 3%, rgba(0, 0, 0, 0) 0), linear-gradient(1turn, rgba(159, 219, 252, 0.15) 3%, rgba(0, 0, 0, 0) 0); background-size: 20px 20px; background-position: center }
.markdown-body h1, .markdown-body h2, .markdown-body h3, .markdown-body h4, .markdown-body h5, .markdown-body h6 { padding: 30px 0; margin-top: 35px; margin-bottom: 10px; color: rgba(77, 208, 225, 1) }
.markdown-body h1 { font-size: 30px; text-align: center; position: relative; width: max-content; margin: 0 auto }
.markdown-body h1:before { position: absolute; content: ""; z-index: -1; top: -20px; height: 100%; width: 100px; left: 0; right: 0; margin: 0 auto; background: url("") center / 64px 64px no-repeat; opacity: 0.84 }
.markdown-body h1:after { position: absolute; content: ""; width: 150%; left: -25%; height: 50%; bottom: 12px; border-radius: 50%; background: linear-gradient(rgba(0, 0, 0, 0) 80%, rgba(77, 208, 225, 0.8)); opacity: 0.6; animation: 6s linear infinite h1animate }
@keyframes h1Animate { 0% { background-position: right bottom } 50% { background-position: right } 100% { background-position: right bottom } }
.markdown-body h2 { display: block; border-bottom: 4px solid rgba(77, 208, 225, 1); position: relative; font-size: 24px; padding: 12px 32px; margin: 30px 0 }
.markdown-body h2:before { width: 24px; height: 24px; left: 0; top: 0; margin: auto; background-size: 24px 24px; background-image: url("") }
.markdown-body h2:after, .markdown-body h2:before { content: ""; display: block; position: absolute; bottom: 0 }
.markdown-body h2:after { right: 0; width: 400px; height: 10px; border-top-right-radius: 24px; background: linear-gradient(90deg, rgba(255, 255, 255, 1), rgba(77, 208, 225, 1)); max-width: 50vw }
.markdown-body h3 { margin: 30px 0; font-size: 18px; position: relative; padding: 4px 32px; width: max-content }
.markdown-body h3:before { border-bottom: 2px solid rgba(77, 208, 225, 1); width: 100%; content: ""; display: block; height: 28px; position: absolute; left: 0; top: 0; bottom: -2px; margin: auto; background-size: 28px 28px; background-image: url(""); background-repeat: no-repeat; animation: 2s infinite alternate h3animationbefore }
@keyframes h3AnimationBefore { 0% { width: 28px } 25% { width: 100% } 50% { width: 100% } 100% { width: 100% } }
.markdown-body h3:after { content: ""; display: block; width: 28px; height: 28px; position: absolute; border: 2px solid rgba(77, 208, 225, 1); border-radius: 50%; right: -15px; top: 0; bottom: 0; margin: auto; background-size: 28px 28px; background-image: url(""); animation: 2s infinite alternate h3animationafter }
@keyframes h3AnimationAfter { 0% { } 10% { } 50% { transform: rotate(-1turn) } 100% { transform: rotate(-1turn) } }
.markdown-body h4 { font-size: 16px }
.markdown-body h5 { font-size: 15px }
.markdown-body h6 { margin-top: 5px }
.markdown-body p { line-height: inherit; margin: 22px 0; letter-spacing: 2px; font-size: 14px; word-spacing: 2px }
.markdown-body img { max-width: 80%; border-radius: 6px; display: block; margin: 20px auto !important; object-fit: contain; box-shadow: 0 0 16px rgba(110, 110, 110, 0.45) }
.markdown-body figcaption { display: block; font-size: 13px; color: rgba(43, 43, 43, 1) }
.markdown-body figcaption:before { content: ""; background-image: url(""); display: inline-block; width: 18px; height: 18px; background-size: 18px; background-repeat: no-repeat; background-position: center; margin-right: 5px; margin-bottom: -5px }
.markdown-body hr { border-top: 1px solid rgba(77, 208, 225, 1); border-right: none; border-bottom: none; border-left: none; margin-top: 32px; margin-bottom: 32px }
.markdown-body del { color: rgba(77, 208, 225, 1) }
.markdown-body code { border-radius: 2px; overflow-x: auto; background-color: rgba(77, 208, 225, 0.08); color: rgba(38, 198, 218, 1); padding: 0.195em 0.4em }
.markdown-body pre { font-family: Menlo, Monaco, Consolas, Courier New, monospace; overflow: auto; position: relative; line-height: 1.75; box-shadow: 0 0 8px rgba(110, 110, 110, 0.45); border-radius: 4px; margin: 16px }
.markdown-body pre:before { content: ""; display: block; height: 30px; width: 100%; margin-bottom: -7px; background: url("") 10px 10px / 40px no-repeat }
.markdown-body pre>code { font-size: 12px; padding: 15px 12px; margin: 0; word-break: normal; display: block; overflow-x: auto; color: rgba(51, 51, 51, 1); background: rgba(248, 248, 248, 1) }
.markdown-body a { color: rgba(77, 208, 225, 1); border-bottom: 1px solid rgba(77, 208, 225, 1); font-weight: 400; text-decoration: none; margin: 0 4px }
.markdown-body a:active, .markdown-body a:hover { background-color: rgba(77, 208, 225, 0.1) }
.markdown-body strong { color: rgba(38, 198, 218, 1) }
.markdown-body strong:before { content: "「" }
.markdown-body strong:after { content: "」" }
.markdown-body em { font-style: normal; color: rgba(77, 208, 225, 1); font-weight: 700 }
.markdown-body table { display: inline-block !important; font-size: 12px; width: auto; max-width: 100%; overflow: auto; border: 1px solid rgba(246, 246, 246, 1) }
.markdown-body thead { background: rgba(246, 246, 246, 1); color: rgba(0, 0, 0, 1); text-align: left }
.markdown-body tr:nth-child(2n) { background-color: rgba(77, 208, 225, 0.05) }
.markdown-body td, .markdown-body th { padding: 12px 7px; line-height: 24px }
.markdown-body td { min-width: 120px }
.markdown-body blockquote { margin: 2em 0; padding: 24px 32px; border-left: 4px solid rgba(38, 198, 218, 1); background: rgba(77, 208, 225, 0.15); position: relative }
.markdown-body blockquote:before { content: "❝"; top: 8px; left: 8px; color: rgba(77, 208, 225, 1); font-size: 30px; line-height: 1; font-weight: 700; position: absolute; opacity: 0.7 }
.markdown-body blockquote:after { content: "❞"; font-size: 30px; position: absolute; right: 8px; bottom: 0; color: rgba(77, 208, 225, 1); opacity: 0.7 }
.markdown-body blockquote p { color: rgba(89, 89, 89, 1); line-height: 2 }
.markdown-body ol, .markdown-body ul { color: rgba(89, 89, 89, 1); padding-left: 28px }
.markdown-body ol li, .markdown-body ul li { margin-bottom: 0; list-style: inherit }
.markdown-body ol li .task-list-item, .markdown-body ul li .task-list-item { list-style: none }
.markdown-body ol li .task-list-item ol, .markdown-body ol li .task-list-item ul, .markdown-body ul li .task-list-item ol, .markdown-body ul li .task-list-item ul { margin-top: 0 }
.markdown-body ol ol, .markdown-body ol ul, .markdown-body ul ol, .markdown-body ul ul { margin-top: 3px }
.markdown-body ol li { padding-left: 6px }
@media (max-width: 720px) { .markdown-body h1 { font-size: 24px } .markdown-body h2 { font-size: 20px } .markdown-body h3 { font-size: 18px } }.markdown-body pre, .markdown-body pre>code.hljs { background: rgba(255, 255, 255, 1); color: rgba(0, 0, 0, 1) }
.hljs-comment, .hljs-quote, .hljs-variable { color: rgba(0, 128, 0, 1) }
.hljs-built_in, .hljs-keyword, .hljs-name, .hljs-selector-tag, .hljs-tag { color: rgba(0, 0, 255, 1) }
.hljs-addition, .hljs-attribute, .hljs-literal, .hljs-section, .hljs-string, .hljs-template-tag, .hljs-template-variable, .hljs-title, .hljs-type { color: rgba(163, 21, 21, 1) }
.hljs-deletion, .hljs-meta, .hljs-selector-attr, .hljs-selector-pseudo { color: rgba(43, 145, 175, 1) }
.hljs-doctag { color: rgba(128, 128, 128, 1) }
.hljs-attr { color: rgba(255, 0, 0, 1) }
.hljs-bullet, .hljs-link, .hljs-symbol { color: rgba(0, 176, 232, 1) }
.hljs-emphasis { font-style: italic }
.hljs-strong { font-weight: 700 }
在 Web 前端开发中,需要将数据动态渲染到页面上,随着应用程序的复杂度增加,数据渲染的逻辑也变得越来越复杂,这时候就需要使用模板引擎来帮助我们动态生成 HTML 标记。本文将深入介绍 JavaScript 模板引擎,帮助读者更好地理解和应用模板引擎。
什么是模板引擎
模板引擎是一种将数据和模板结合起来生成最终结果的工具,它将一个模板和一个数据对象作为输入,通过模板解析和渲染生成最终的结果。通俗地说,模板引擎就是用来生成 HTML 等标记的工具。
为什么需要模板引擎
在 Web 开发中,数据的呈现通常是基于 HTML 和 CSS 的,而数据的变化又是非常频繁的,需要根据数据动态生成 HTML 标记。手动拼接 HTML 标记显然是一种非常低效的方式,不仅容易出错,而且难以维护。这时候就需要使用模板引擎来完成面向数据的标记生成。
前端模板引擎的分类
JavaScript 模板引擎通常分为以下几类:
- 原生字符串拼接
- 字符串模板
- 函数式模板
- 编译型模板
原生字符串拼接
最早期的模板引擎实现方式是通过字符串拼接来生成需要的 HTML,这种实现方式通常需要手动拼接 HTML 标签,非常繁琐,同时也容易出错。下面是一个使用原生字符串拼接实现模板引擎的示例:
function render(data) {
var html = "<ul>";
for (var i = 0; i < data.length; i++) {
html += "<li>" + data[i].name + "</li>";
}
html += "</ul>";
return html;
}
这种实现方式的缺点是难以维护,但是优点是非常简单。
字符串模板
字符串模板通过字符串替换来实现模板引擎,它可以在字符串中插入变量和表达式,因此字符串模板看起来更加直观和易于维护。下面是一个基于字符串模板的简单模板引擎实现示例:
function render(data) {
var tpl = "<ul>";
for (var i = 0; i < data.length; i++) {
tpl += "<li>{{name}}</li>".replace("{{name}}", data[i].name);
}
tpl += "</ul>";
return tpl;
}
这种实现方式相比于原生字符串拼接更加直观和易于维护。
函数式模板
函数式模板通常是通过函数的调用来生成需要的 HTML,函数接受一个数据对象作为参数,并返回一个代表 HTML 的字符串。这种方式通常使用字符串模板来实现,但是相比于字符串模板更加灵活。下面是一个基于函数式模板的示例:
function listItem(data) {
return "<li>" + data.name + "</li>";
}
function render(data) {
var html = "<ul>" + data.map(listItem).join("") + "</ul>";
return html;
}
这种实现方式的优点是灵活,可以通过函数实现复杂的逻辑,同时也比字符串模板更加易于维护。
编译型模板
编译型模板通常将模板转换成一个 JavaScript 函数,这个函数接受一个数据对象作为参数,并返回一个代表 HTML 的字符串。这种方式的优点是性能好,而且可以预编译模板,提高渲染效率。下面是一个基于编译型模板的示例:
var tpl = "<ul>{{#each data}}<li>{{name}}</li>{{/each}}</ul>";
var compiled = Handlebars.compile(tpl);
function render(data) {
return compiled({ data: data });
}
这种实现方式的缺点是需要引入编译器,同时也需要一定的学习成本。
JavaScript 模板引擎的实现原理
模板解析
模板引擎通常需要将模板解析成一个 AST(抽象语法树),这个 AST 会记录模板中的变量、表达式和文本等信息,同时也可以用来验证模板的语法正确性。模板解析通常是一个比较复杂的过程,需要处理一些复杂的语法结构(如条件语句、循环语句等),同时还需要支持一些扩展特性(如过滤器、自定义标签等)。
模板渲染
模板渲染通常是一个根据数据和 AST 生成 HTML 的过程,它需要遍历 AST,并将 AST 中的变量和表达式替换成实际的值。模板渲染通常需要考虑一些细节问题(如文本转义、属性值处理等),同时也需要支持一些扩展特性(如数据格式化、事件绑定等)。
常见的 JavaScript 模板引擎
目前比较流行的 JavaScript 模板引擎有 Mustache、EJS、Jade/Pug、Handlebars 和 Vue Template 等。
Mustache
Mustache 是一种基于字符串模板的模板引擎,它支持数据绑定、条件语句、循环语句等语法结构,同时也支持自定义标签和过滤器等扩展特性。下面是一个基于 Mustache 的示例:
var tpl = "<ul>{{#data}}<li>{{name}}</li>{{/data}}</ul>";
function render(data) {
return Mustache.render(tpl, { data: data });
}
EJS
EJS 是一种基于字符串模板的模板引擎,它支持数据绑定、条件语句、循环语句等语法结构,同时也支持模板继承和局部模板等特性。下面是一个基于 EJS 的示例:
var tpl = "<ul><% for (var i = 0; i < data.length; i++) { %><li><%= data[i].name %></li><% } %></ul>";
function render(data) {
return ejs.render(tpl, { data: data });
}
Jade/Pug
Jade/Pug 是一种基于缩进的模板引擎,它将 HTML 标记和 JavaScript 代码混合在一起,通过缩进来表示代码的层次结构。Jade/Pug 支持数据绑定、条件语句、循环语句等语法结构,同时也支持模板继承和动态属性等特性。下面是一个基于 Jade/Pug 的示例:
var tpl = "ul\n each data\n li #{this.name}";
function render(data) {
var fn = pug.compile(tpl);
return fn({ data: data });
}
Handlebars
Handlebars 是一种基于编译型模板的模板引擎,它支持数据绑定、条件语句、循环语句等语法结构,同时也支持自定义标签和助手函数等扩展特性。下面是一个基于 Handlebars 的示例:
var tpl = "<ul>{{#each data}}<li>{{name}}</li>{{/each}}</ul>";
var compiled = Handlebars.compile(tpl);
function render(data) {
return compiled({ data: data });
}
Vue Template
Vue Template 是 Vue.js 框架内置的模板引擎,它支持数据绑定、条件语句、循环语句等语法结构,同时也支持计算属性、事件绑定和组件化等扩展特性。下面是一个基于 Vue Template 的示例:
<template>
<ul>
<li v-for="item in data">{{item.name}}</li>
</ul>
</template>
<script>
export default {
props: {
data: { type: Array }
}
}
</script>
<style scoped>
ul { list-style-type: none; }
</style>
如何选择合适的模板引擎
在选择模板引擎时,需要综合考虑以下几个方面:
性能
模板引擎的性能是非常重要的,特别是在处理大量数据时。通常情况下,编译型模板引擎的性能比基于字符串的模板引擎更好,但是编译过程需要一定的时间,因此需要权衡性能和编译时间的关系。
语法
模板引擎的语法也是非常重要的,它需要易于理解和记忆,并且尽可能地接近原生的 HTML 和 JavaScript 语法。同时也需要支持一些扩展特性,以满足复杂的业务需求。
社区支持
模板引擎的社区支持也是非常重要的,它需要有活跃的社区和广泛的使用,以便获取帮助和解决问题。
结语
本文中,我们学习了模板引擎的基本概念和分类,介绍了常见的 JavaScript 模板引擎及其实现原理,并提供了如何选择合适的模板引擎的建议。通过本文的学习,读者可以更加深入地理解和应用模板引擎,在实际项目中提高代码的效率。
深入理解 JavaScript 模板引擎的更多相关文章
- JavaScript 模板引擎实现原理解析
1.入门实例 首先我们来看一个简单模板: <script type="template" id="template"> <h2> < ...
- Javascript模板引擎mustache.js详解
mustache.js是一个简单强大的Javascript模板引擎,使用它可以简化在js代码中的html编写,压缩后只有9KB,非常值得在项目中使用.本文总结它的使用方法和一些使用心得,内容不算很高深 ...
- 【JavsScript】推荐五款流行的JavaScript模板引擎
摘要:Javascript模板引擎作为数据与界面分离工作中最重要一环,受到开发者广泛关注.本文通过开发实例解析五款流行模板引擎:Mustache.Underscore Templates.Embedd ...
- 推荐13款javascript模板引擎
javaScript 在生成各种页面内容时如果能结合一些模板技术,可以让逻辑和数据之间更加清晰,本文介绍 X 款 JavaScript 的模板引擎.(排名不分先后顺序) 1. Mustache 基于j ...
- 各种JS模板引擎对比数据(高性能JavaScript模板引擎)
最近做了JS模板引擎测试,拿各个JS模板引擎在不同浏览器上去运行同一程序,下面是模板引擎测试数据:通过测试artTemplate.juicer与doT引擎模板整体性能要有绝对优势: js模板引擎 Ja ...
- JavaScript模板引擎artTemplate.js——为什么使用模板引擎?
作为一个工作一年的菜鸟,在公司做了几个外包项目,也接触到了不同形式的web开发.其实也没多少,就是javaweb开发和HTML5移动开发,这两者在页面展示的时候的解决方案还是有所不同的. 1.vo+e ...
- 最简单的JavaScript模板引擎
在小公司待久了感觉自己的知识面很小,最近逛博客园和一些技术网站看大家在说JavaScript模版引擎的事儿,完全没有概念,网上一搜这是08年开始流行起来的...本来以为这是很高深的知识,后来在网上看到 ...
- JavaScript模板引擎实例应用
在之前的一篇名为<移动端基于HTML模板和JSON数据的JavaScript交互>的文章中,我向大家说明了为什么要使用JavaScript模板以及如何使用,文末还提到了laytpl.art ...
- 如何选择Javascript模板引擎(javascript template engine)?
译者 jjfat 日期:2012-9-17 来源: GBin1.com 随着前端开发的密集度越来越高,Ajax和JSON的使用越来越频繁,大家肯定免不了在前台开发中大量的使用标签,常见到的例子如下: ...
- Juicer javascript 模板引擎
模板引擎是为了使用户界面与业务数据(内容)分离而产生的,它可以生成特定格式的文档,用于网站的模板引擎就会生成一个标准的HTML文档. js模板引擎包括如下:template官方参考:http://au ...
随机推荐
- CH340区别
CH340区别 CH340G USB转串⼝,推出时间最早,需外挂晶振,应⽤最⼴SOP16 CH340C USB转串⼝,内置晶振,引脚兼容CH340G SOP16 CH340E USB转串⼝,内置 ...
- wikidata介绍和查询
Wikidata是一个大型结构化开源知识图,为维基百科等项目提供支持.我们可使用SPARQL(Wikidata官方Tutorial)对其进行查询.SPARQL是一种专为 RDF(Resource ...
- 错误修正记录:对应的VMware Tools脚本未能成功运行
起因 装了台式机,想把笔记本里的vmware虚拟机迁移过来:复制过来后开机就出现这种情况,点开机.挂起.关机等操作会报错,然后无法挂起(再点一次就行) 或者使用下方的选项,而非上方包装过的功能,似乎也 ...
- React从webpack迁移到rsbuild 纪实
Why 随着团队项目规模越来越大之后,继从babel-loader迁移到esbuild之后发现打包.热重载性能随着时间迭代之后又慢慢开始成为性能瓶颈,所以决定用新的打包工具去解决这个问题.esbuil ...
- VMware网络虚拟化介绍(之一)
2014年5月,在我加入VMware三个月之后,我涂鸦了一篇<扒一扒SDN的那些事儿>,当时放言如果阅读量过百就写续篇.后来果然阅读量没过百,也就80多的样子,其中好几份还是我自恋地进去查 ...
- 如何打造你自己的 AI 软件工程师(像 Devin 那样)
扩展 DeepSeek 的强化学习蓝图路线到AI的其他方面 Nikhil Anand 图片由GPT-4o生成 "AI 软件工程师"这个概念,其实已经不再遥远了.已经有一些技术在逐步 ...
- 使用Shader画常见的数学函数
使用Shader画常见的数学函数 本篇博文的灵感来自于Shader Books这一小节:https://thebookofshaders.com/05/?lan=ch 代码运行网站:http://ed ...
- 【Docker】简介
Docker 简介 某个应用,如果可以提供服务,那么就可以打包成docker供给他人使用 是什么 我们具体来看看Docker. 大家需要注意,Docker本身并不是容器,它是创建容器的工具,是应用容器 ...
- 如何查看 linux 发行版本
以 debian 10 buster 为例 有时候我们需要知道当前正在使用的 linux 的发行版本信息...可以通过下面几种方式来查看 使用 lsb_release 命令查看 lsb_release ...
- golang的条件编译
写c/c++或者rust的开发者应该对条件编译不陌生,条件编译顾名思义就是在编译时让代码中的一部分生效或者失效,从而控制编译时的代码执行路径,进而影响编译出来的程序的行为. 这有啥用呢?通常在编写跨平 ...