iis服务器助手广告广告
返回顶部
首页 > 资讯 > 精选 >怎么用JavaScript写一款EJS模板引擎
  • 942
分享到

怎么用JavaScript写一款EJS模板引擎

2023-06-29 04:06:10 942人浏览 独家记忆
摘要

本篇内容介绍了“怎么用javascript写一款Ejs模板引擎”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!1. 起因部门最近的一次分享中,

本篇内容介绍了“怎么用javascript写一款Ejs模板引擎”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!

1. 起因

部门最近的一次分享中,有人提出来要实现一个ejs模板引擎,突然发现之前似乎从来都没有考虑过这个问题,一直都是直接拿过来用的。那就动手实现一下吧。本文主要介绍ejs的简单使用,并非全部实现,其中涉及到options配置的部分直接省略了。如有不对请指出,最后欢迎点赞 + 收藏。

2. 基本语法实现

定义render函数,接收html字符串,和data参数。

const render = (ejs = '', data = {}) => {}

事例模板字符串如下:

<body>    <div><%= name %></div>    <div><%= age %></div></body>

可以使用正则将<%= name %>匹配出来,只保留name。这里借助es6的模板字符串。将name用${}包裹起来。

props中第2个值就是匹配到的变量。直接props[1]替换。

[  '<%= name %>',  ' name ',  16,  '<body>\n    <div><%= name %></div>\n    <div><%= age %></div>\n</body>']
const render = (ejs = '', data = {}) => {    const html = ejs.replace(/<%=(.*?)%>/g, (...props) => {        return '${' + props[1] + '}';        // return data[props[1].trim()];    });}

3. Function函数

这里得到的html是一个模板字符串。可以通过Function将字符串编程可执行的函数。当然这里也可以使用eval,随你。

<body>    <div>${ name }</div>    <div>${ age }</div></body>

Function是一个构造函数,实例化后返回一个真正的函数,构造函数的最后一个参数是函数体的字符串,前面的参数都为形式参数。比如这里传入形参name,函数体通过console.log打印一句话。

const func = new Function('name', 'console.log("我是通过Function构建的函数,我叫:" + name)');// 执行函数,传入参数func('yindong'); // 我是通过Function构建的函数,我叫:yindong

利用Function的能力可以将html模板字符串执行返回。函数字符串编写return,返回一个拼装好的模板字符串、

const getHtml = (html, data) => {    const func = new Function('data', `return \`${html}\`;`);    return func(data);    // return eval(`((data) => {  return \`${html}\`; })(data)`)}const render = (ejs = '', data = {}) => {    const html = ejs.replace(/<%=(.*?)%>/g, (...props) => {        return '${' + props[1] + '}';    });    return getHtml(html, data);}

4 with

这里render函数中props[1]的实际上是变量名称,也就是name和age,可以替换成data[props[1].trim()],不过这样写会有一些问题,偷个懒利用with代码块的特性。

with语句用于扩展一个语句的作用域链。换句人话来说就是在with语句中使用的变量都会先在with中寻找,找不到才会向上寻找。

比如这里定义一个age数字和data对象,data中包含一个name字符串。with包裹的代码块中输出的name会先在data中寻找,age在data中并不存在,则会向上寻找。当然这个特性也是一个with不推荐使用的原因,因为不确定with语句中出现的变量是否是data中。

const age = 18;const data = {    name: 'yindong'}with(data) {    console.log(name);    console.log(age);}

这里使用with改造一下getHtml函数。函数体用with包裹起来,data就是传入的参数data,这样with体中的所有使用的变量都从data中查找了。

const getHtml = (html, data) => {    const func = new Function('data', `with(data) { return \`${html}\`; }`);    return func(data);    // return eval(`((data) => { with(data) { return \`${html}\`; } })(data)`)}const render = (ejs = '', data = {}) => {    // 优化一下代码,直接用$1替代props[1];    // const html = ejs.replace(/<%=(.*?)%>/g, (...props) => {    //     return '${' + props[1] + '}';    // });    const html = ejs.replace(/<%=(.*?)%>/gi, '${$1}');    return getHtml(html, data);}

这样就可以打印出真是的html了。

<body>    <div>yindong</div>    <div>18</div></body>

5. ejs语句

这里扩展一下ejs,加上一个arr.join语句。

<body>    <div><%= name %></div>    <div><%= age %></div>    <div><%= arr.join('--') %></div></body>
const data = {    name: "yindong",    age: 18,    arr: [1, 2, 3, 4]}const html = fs.readFileSync('./html.ejs', 'utf-8');const getHtml = (html, data) => {    const func = new Function('data', ` with(data) { return \`${html}\`; }`);    return func(data);}const render = (ejs = '', data = {}) => {    const html = html = ejs.replace(/<%=(.*?)%>/gi, '${$1}');    return getHtml(html, data);}const result = render(html, data);console.log(result);

可以发现ejs也是可以正常编译的。因为模板字符串支持arr.join语法,输出:

<body>    <div>yindong</div>    <div>18</div>    <div>1--2--3--4</div></body>

如果ejs中包含forEach语句,就比较复杂了。此时render函数就无法正常解析。

<body>    <div><%= name %></div>    <div><%= age %></div>    <% arr.forEach((item) => {%>        <div><%= item %></div>    <%})%></body>

这里分两步来处理。仔细观察可以发现,使用变量值得方式存在=号,而语句是没有=号的。可以对ejs字符串进行第一步处理,将<%=变量替换成对应的变量,也就是原本的render函数代码不变。

const render = (ejs = '', data = {}) => {    const html = ejs.replace(/<%=(.*?)%>/gi, '${$1}');    console.log(html);}
<body>    <div>${ name }</div>    <div>${ age }</div>    <% arr.forEach((item) => {%>        <div>${ item }</div>    <%})%></body>

第二步比较绕一点,可以将上面的字符串处理成多个字符串拼接。简单举例,将a加上arr.forEach的结果再加上c转换为,str存储a,再拼接arr.forEach每项结果,再拼接c。这样就可以获得正确的字符串了。

// 原始字符串retrun `    a    <% arr.forEach((item) => {%>        item    <%})%>    c`// 拼接后的let str;str = `a`;arr.forEach((item) => {    str += item;});str += c;return str;


在第一步的结果上使用/<%(.*?)%>/g正则匹配出<%%>中间的内容,也就是第二步。

const render = (ejs = '', data = {}) => {    // 第一步    let html = ejs.replace(/<%=(.*?)%>/gi, '${$1}');    // 第二步    html = html.replace(/<%(.*?)%>/g, (...props) => {        return '`\r\n' + props[1] + '\r\n str += `';    });    console.log(html);}

替换后得到的字符串长成这个样子。

<body>    <div>${ name }</div>    <div>${ age }</div>    ` arr.forEach((item) => { str += `        <div>${ item }</div>    `}) str += `</body>

添加换行会更容易看一些。可以发现,第一部分是缺少首部`的字符串,第二部分是用str存储了forEach循环内容的完整js部分,并且可执行。第三部分是缺少尾部`的字符串。

<body>    <div>${ name }</div>    <div>${ age }</div>    `// 第二部分 arr.forEach((item) => { str += `        <div>${ item }</div>    `})// 第三部分 str += `</body>

处理一下将字符串补齐,在第一部分添加let str = `,这样就是一个完整的字串了,第二部分不需要处理,会再第一部分基础上拼接上第二部分的执行结果,第三部分需要在结尾出拼接`; return str; 也就是补齐尾部的模板字符串,并且通过return返回str完整字符串。

// 第一部分let str = `<body>    <div>${ name }</div>    <div>${ age }</div>    `// 第二部分 arr.forEach((item) => { str += `        <div>${ item }</div>    `})// 第三部分 str += `</body>`;return str;

这部分逻辑可以在getHtml函数中添加,首先在with中定义str用于存储第一部分的字符串,尾部通过return返回str字符串。

const getHtml = (html, data) => {    const func = new Function('data', ` with(data) { let str = \`${html}\`; return str; }`);    return func(data);}

这样就可以实现执行ejs语句了。

const data = {    name: "yindong",    age: 18,    arr: [1, 2, 3, 4],    html: '<div>html</div>',    escape: '<div>escape</div>'}const html = fs.readFileSync('./html.ejs', 'utf-8');const getHtml = (html, data) => {    const func = new Function('data', ` with(data) { var str = \`${html}\`; return str; }`);    return func(data);}const render = (ejs = '', data = {}) => {    // 替换所有变量    let html = ejs.replace(/<%=(.*?)%>/gi, '${$1}');    // 拼接字符串    html = html.replace(/<%(.*?)%>/g, (...props) => {        return '`\r\n' + props[1] + '\r\n str += `';    });    return getHtml(html, data);}const result = render(html, data);console.log(result);

输出结果:

<body>
    <div>yindong</div>
    <div>18</div>

        <div>1</div>

        <div>2</div>

        <div>3</div>

        <div>4</div>

</body>

6. 标签转义

<%=会对传入的html进行转义,这里编写一个escapeHTML转义函数。

const escapeHTML = (str) => {    if (typeof str === 'string') {        return str.replace(/&/g, "&amp;").replace(/</g, "&lt;").replace(/>/g, "&gt;").replace(/ /g, "&nbsp;").replace(/"/g, "&#34;").replace(/'/g, "&#39;");    } else {        return str;    }}

变量替换的时候使用escapeHTML函数处理变量。这里通过\s*去掉空格。为了避免命名冲突,这里将escapeHTML改造成自执行函数,函数参数为$1变量名。

const render = (ejs = '', data = {}) => {    // 替换转移变量    // let html = ejs.replace(/<%=\s*(.*?)\s*%>/gi, '${escapeHTML($1)}');    let html = ejs.replace(/<%=\s*(.*?)\s*%>/gi, `\${        ((str) => {            if (typeof str === 'string') {                return str.replace(/&/g, "&amp;").replace(/</g, "&lt;").replace(/>/g, "&gt;").replace(/ /g, "&nbsp;").replace(/"/g, "&#34;").replace(/'/g, "&#39;");            } else {                return str;            }        })($1)    }`);    // 拼接字符串    html = html.replace(/<%(.*?)%>/g, (...props) => {        return '`\r\n' + props[1] + '\r\n str += `';    });    return getHtml(html, data);}

getHtml函数不变。

const getHtml = (html, data) => {    const func = new Function('data', `with(data) { var str = \`${html}\`; return str; }`);    return func(data);}

<%-会保留原本格式输出,只需要再加一条不使用escapeHTML函数处理的就可以了。

const render = (ejs = '', data = {}) => {    // 替换转义变量    let html = ejs.replace(/<%=\s*(.*?)\s*%>/gi, '${escapeHTML($1)}');    // 替换其余变量    html = html.replace(/<%-(.*?)%>/gi, '${$1}');    // 拼接字符串    html = html.replace(/<%(.*?)%>/g, (...props) => {        return '`\r\n' + props[1] + '\r\n str += `';    });    return getHtml(html, data, escapeHTML);}

输出样式:

<body>
    <div>yindong</div>
    <div>18</div>

        <div>1</div>

        <div>2</div>

        <div>3</div>

        <div>4</div>

    <div>&lt;div&gt;escapeHTML&lt;/div&gt;</div>
</body>

至此一个简单的ejs模板解释器就写完了。

“怎么用JavaScript写一款EJS模板引擎”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识可以关注编程网网站,小编将为大家输出更多高质量的实用文章!

--结束END--

本文标题: 怎么用JavaScript写一款EJS模板引擎

本文链接: https://www.lsjlt.com/news/322489.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • c++中if elseif使用规则
    c++ 中 if-else if 语句的使用规则为:语法:if (条件1) { // 执行代码块 1} else if (条件 2) { // 执行代码块 2}// ...else ...
    99+
    2024-05-14
    c++
  • c++中的继承怎么写
    继承是一种允许类从现有类派生并访问其成员的强大机制。在 c++ 中,继承类型包括:单继承:一个子类从一个基类继承。多继承:一个子类从多个基类继承。层次继承:多个子类从同一个基类继承。多层...
    99+
    2024-05-14
    c++
  • c++中如何使用类和对象掌握目标
    在 c++ 中创建类和对象:使用 class 关键字定义类,包含数据成员和方法。使用对象名称和类名称创建对象。访问权限包括:公有、受保护和私有。数据成员是类的变量,每个对象拥有自己的副本...
    99+
    2024-05-14
    c++
  • c++中优先级是什么意思
    c++ 中的优先级规则:优先级高的操作符先执行,相同优先级的从左到右执行,括号可改变执行顺序。操作符优先级表包含从最高到最低的优先级列表,其中赋值运算符具有最低优先级。通过了解优先级,可...
    99+
    2024-05-14
    c++
  • c++中a+是什么意思
    c++ 中的 a+ 运算符表示自增运算符,用于将变量递增 1 并将结果存储在同一变量中。语法为 a++,用法包括循环和计数器。它可与后置递增运算符 ++a 交换使用,后者在表达式求值后递...
    99+
    2024-05-14
    c++
  • c++中a.b什么意思
    c++kquote>“a.b”表示对象“a”的成员“b”,用于访问对象成员,可用“对象名.成员名”的语法。它还可以用于访问嵌套成员,如“对象名.嵌套成员名.成员名”的语法。 c++...
    99+
    2024-05-14
    c++
  • C++ 并发编程库的优缺点
    c++++ 提供了多种并发编程库,满足不同场景下的需求。线程库 (std::thread) 易于使用但开销大;异步库 (std::async) 可异步执行任务,但 api 复杂;协程库 ...
    99+
    2024-05-14
    c++ 并发编程
  • 如何在 Golang 中备份数据库?
    在 golang 中备份数据库对于保护数据至关重要。可以使用标准库中的 database/sql 包,或第三方包如 github.com/go-sql-driver/mysql。具体步骤...
    99+
    2024-05-14
    golang 数据库备份 mysql git 标准库
  • 如何在 Golang 中优雅地处理错误?
    在 go 中,优雅处理错误包括:使用 error 类型;使用 errors 包函数和类型;自定义错误类型;遵循错误处理模式,包括关闭资源、检查错误、打印错误信息和处理或返回错误。 在 ...
    99+
    2024-05-14
    golang 错误处理
  • 如何构建 Golang RESTful API,并使用中间件进行身份验证?
    本文介绍了如何构建 golang restful api。首先,通过导入必要的库、定义数据模型和创建路由来构建 restful api。其次,使用 go-chi/chigot 和 go-...
    99+
    2024-05-14
    golang git
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作