关于模板,我倒是用过了不少。最开始要数Java的JSP了,然后接触了PHP的smarty,再就是Python的jinja2, Django内置模板,现在刚开始看Nodejs,也发现了不少类似的模板引擎,ejs, jade等等吧。

模板带来的最直接的好处就是加速开发,前后端分离。除此之外,对于字符串的格式化同样是个比较好的应用。习惯了python中
string = "hello {}".format("郭璞") # hello 郭璞
string = "hello {username}".format(username="郭璞") # hello 郭璞
这样简便的用法,突然来到nodejs中,没有了这类特性的原生支持,写起来打印语句就老是觉得很别扭,一点都不优雅。然后我就想自己做一个实现上述功能的工具函数,方便自己的使用。然后就想到了模板这一个方向,虽然想法还不够成熟,甚至是有点拙略,但是“灵(瞎)感(闹)”还是得记录一下不是。
Function对象
JavaScript中有这么一个神奇的对象,那就是Function。如果函数体符合语法要求,那么你就可以动态创建出一个自己的函数出来。下面来个简单的小例子。
无参模式
function create_function(){
var func_body = "var time = new Date(); console.log('创建时间:'+time);";
var func = new Function('', func_body);
func();
}
create_function();
运行结果如下:
E:\Code\Nodejs\learn\my-work\string>node one.js 创建时间:Tue Jun 13 2017 15:40:15 GMT+0800 (中国标准时间) E:\Code\Nodejs\learn\my-work\string>
有参模式
刚才演示了一个无参数的情况,那么有参数的情况如何呢?
function create_function_with_parameters() {
var param1 = "郭璞";
var param2 = "辽宁大连";
var func_body = "console.log('Hello '+param1+', welcome to '+param2+'!' );";
var func = new Function('param1', 'param2', func_body);
func(param1, param2);
}
create_function_with_parameters();
同样的运行结果如下:
E:\Code\Nodejs\learn\my-work\string>node one.js Hello 郭璞, welcome to 辽宁大连! E:\Code\Nodejs\learn\my-work\string>
到这里,关于Function的内容就算是铺垫完成了。只需要了解这
正则
探究模板的真实原理,有些语言中是编译型的,有些是替换型的。但是不管是哪种类型,都离不开扣出变量关键字这个步骤。而这个过程用正则表达式基本上是最好的方法了。所以需要掌握一点相关的技巧。
如何表达?
在Nodejs中,使用正则表达式有两种形式:
关于正则表达式的具体的规则,鉴于篇幅很长,这里就不再赘述了。有兴趣的可以浏览下面的这篇文章。
//www./article/39623.htm?source=1
需求获取
根据一开始的设想,目标是获取{{}} 和{%%} 这种语法下的变量名称,然后替换成对应的变量值。 因此可以写出如下的正则表达式:
var pattern1 = /{{([\s\S]+?)}}/gi;
// 或者
var pattern2 = /{%([\s\S]+?)%}/gi;
默认规则如下:
简易实现
下面简单的对照着实现一下。
直接变量形式
function test1(){
var tpl = "Hello {{visitorname}}, Welcome to {{worldname}}!";
var data = {
visitorname: "游客",
worldname: "冰雹工作室"
};
var pattern = /{{([\s\S]+?)}}/gi;
var result = tpl.replace(pattern, (match, tuple)=>{
return data[tuple];
});
console.log("渲染后的数据为:\n", result);
}
实现结果:
E:\Code\Nodejs\learn\my-work\string>node one.js 渲染后的数据为: Hello 游客, Welcome to 冰雹工作室! E:\Code\Nodejs\learn\my-work\string>
对象形式
function test2(){
var tpl = "I'm {{user.name}}, and I come from {{user.address}}";
var user = {name: "郭璞", address: "辽宁大连"};
console.log(user.name);
var pattern = /{{([\s\S]+?)}}/gi;
var result = tpl.replace(pattern, function(match, tuple, offset){
return eval(''+tuple);
});
console.log(result);
}
运行效果:
E:\Code\Nodejs\learn\my-work\string>node one.js 郭璞 I'm 郭璞, and I come from 辽宁大连 E:\Code\Nodejs\learn\my-work\string>
混杂多参数实现
刚才实现了只有关键字的和有对象性质的参数的例子,但是实际中情况可能比这要复杂的多,比如混杂模式。接下来着手实现一下混杂模式下的替换策略。
function test3(){
var tpl = "I am {} of {} years old, and I come from {user.address}.";
var name = '郭璞';
var index = 0;
var paramindex = 0;
// var parameters = [{name: '郭璞'}, {'age': 22}, {address: '辽宁大连'}];
var parameters = ['郭璞', 22, {user: {address: '辽宁大连'}}];
console.log(parameters[2]);
var result = tpl.replace(/{([\s\S])*?}/gi, function(match, tuple, offset){
console.log('match:', match);
console.log('tuple: ', tuple);
tpl = tpl.slice(index, offset);
index = offset + match.length;
paramindex += 1;
var temp = parameters[paramindex-1];
if(match.length > 2){
// 使用tuple不能正确获取到标记中相关的变量名,故用match来代替.
match = match.slice(1, match.length-1);
return eval('parameters[paramindex-1].'+match);
}else{
return temp;
}
// return parameters[paramindex-1];
});
console.log(result);
}
运行结果如下:
E:\Code\Nodejs\learn\my-work\string>node one.js
{ user: { address: '辽宁大连' } }
match: {}
tuple: undefined
match: {}
tuple: undefined
match: {user.address}
tuple: s
******* s
I am 郭璞 of 22 years old, and I come from 辽宁大连.
E:\Code\Nodejs\learn\my-work\string>
关于正则这块,大致的内容就是这样了。如果要想更简单的调用,只需要封装起来,用外部参数代替就好了。
当然,注意变量名的命名风格。
实战
废话连篇说了两个小节,还没到正式的模板制作。下面就整合一下刚才例子。模拟着实现一下好了。
(!完整)代码
来个不完整的代码,示意一下算了。
/**
* 通过正则表达式和Function语法创建一个简单的模板引擎。
*/
const pattern = /{{([\s\S]+?)}}|{%([\s\S]+?)%}|$/img;
function template(text, params, name) {
// 声明最终要返回的解析好的文本串,也就是构造Function所需的函数体部分。
var func_body = '';
// 函数体里面最终效果是返回一个代表了解析完成的字符串的变量,因此要声明一个出来
func_body += 'var parsedstr="";';
func_body += 'parsedstr+="';
var index = 0;
// 开始正则匹配,根据捕获到的元组进行剖析
text.replace(pattern, function (matchedtext, interpolate, evaluate, offset) {
// 匹配到正常的HTML文本,则直接添加到func_body中即可
func_body += text.slice(index, offset);
// 如果是evaluate类型的文本,则作为代码进行拼接
if (evaluate) {
func_body += '";' + evaluate + 'parsedstr+="';
}
// 匹配到interpolate类型的文本,则作为变量值进行替换
if (interpolate) {
func_body += '"+' + interpolate + '+"';
}
// 更新偏移量index,让程序向后移动
index = offset + matchedtext.length;
// 貌似返回值没什么用吧
return matchedtext;
});
// 完成函数体的构建之后就可以调用Function的语法实现渲染函数的构建了
func_body += '"; return parsedstr;';
return new Function('obj', 'name', func_body)(params, name);
}
function test() {
var obj = [
{ text: '张三' },
{ text: '李四' },
{ text: '王五' },
{ text: '赵六' },
{ text: '韩七' },
{ text: '王八' }
];
var name = '郭璞';
var fs = require('fs');
// var rawtext = fs.readFileSync('index.html').toString('utf8');
var rawtext = '<ul>{%for(var i in obj){%}<li>{{ obj[i].text }}</li>
{%}%}</ul>';
console.log("源文件:", rawtext);
var result = template(rawtext, obj);
console.log("渲染后文件:", result, name);
fs.writeFileSync('rendered.html', result);
console.log('渲染完毕,请查看rendered.html文件')
}
test();
同级目录下生成的文件内容为:
<ul> <li>张三</li><br> <li>李四</li><br> <li>王五</li><br> <li>赵六</li><br> <li>韩七</li><br> <li>王八</li><br></ul>
感觉效果还行,但是这里面参数太固定化了,实际封装的时候还需要酌情指定,不然这东西也就没什么用。
总结
要是论实用性价值的话,这个不成熟的模板实现思路毫无价值。但是对于我而言,用来格式化字符串倒是个不错的选择,估计我会把这个小思路封装成一个小小的模块,详情https://github.com/guoruibiao/have-fun-in-node
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
# nodejs模板引擎
# node.js 模板引擎
# nodejs模板引擎制作
# 自己的
# 是个
# 正则表达式
# 变量名
# 只需要
# 来个
# 李四
# 这一
# 废话连篇
# 王五
# 好了
# 都不
# 变量值
# 说了
# 则是
# 中有
# 所需
# 就没
# 这类
# 要想
相关文章:
网站按钮制作软件,如何实现网页中按钮的自动点击?
湖北网站制作公司有哪些,湖北清能集团官网?
如何彻底删除建站之星生成的Banner?
Avalonia如何实现跨窗口通信 Avalonia窗口间数据传递
建站主机选购指南与交易推荐:核心配置解析
阿里云高弹*务器配置方案|支持分布式架构与多节点部署
如何确保西部建站助手FTP传输的安全性?
建站之星导航如何优化提升用户体验?
家族网站制作贴纸教程视频,用豆子做粘帖画怎么制作?
建站之星如何保障用户数据免受黑客入侵?
北京营销型网站制作公司,可以用python做一个营销推广网站吗?
成都网站制作价格表,现在成都广电的单独网络宽带有多少的,资费是什么情况呢?
网页设计网站制作软件,microsoft office哪个可以创建网页?
如何配置支付宝与微信支付功能?
做企业网站制作流程,企业网站制作基本流程有哪些?
如何选择香港主机高效搭建外贸独立站?
建站之星收费标准详解:套餐费用及年费价格表一览
建站之星安装提示数据库无法连接如何解决?
如何通过万网虚拟主机快速搭建网站?
如何用PHP快速搭建CMS系统?
保定网站制作方案定制,保定招聘的渠道有哪些?找工作的人一般都去哪里看招聘信息?
专业的网站制作设计是什么,如何制作一个企业网站,建设网站的基本步骤有哪些?
如何制作一个表白网站视频,关于勇敢表白的小标题?
国美网站制作流程,国美电器蒸汽鍋怎么用官方网站?
香港服务器建站指南:外贸独立站搭建与跨境电商配置流程
如何自定义建站之星网站的导航菜单样式?
深圳企业网站制作设计,在深圳如何网上全流程注册公司?
建站之星五站合一营销型网站搭建攻略,流量入口全覆盖优化指南
婚礼视频制作网站,学习*后期制作的网站有哪些?
家庭服务器如何搭建个人网站?
建站之星下载版如何获取与安装?
网站制作难吗安全吗,做一个网站需要多久时间?
建站VPS推荐:2025年高性能服务器配置指南
小型网站制作HTML,*游戏网站怎么搭建?
整蛊网站制作软件,手机不停的收到各种网站的验证码短信,是手机病毒还是人为恶搞?有这种手机病毒吗?
如何正确选择百度移动适配建站域名?
如何通过FTP空间快速搭建安全高效网站?
如何在七牛云存储上搭建网站并设置自定义域名?
如何在阿里云虚拟主机上快速搭建个人网站?
购物网站制作公司有哪些,哪个购物网站比较好?
网站广告牌制作方法,街上的广告牌,横幅,用PS还是其他软件做的?
网站制作公司广州有几家,广州尚艺美发学校网站是多少?
如何选择PHP开源工具快速搭建网站?
制作充值网站的软件,做人力招聘为什么要自己交端口钱?
青岛网站建设如何选择本地服务器?
开心动漫网站制作软件下载,十分开心动画为何停播?
如何通过WDCP绑定主域名及创建子域名站点?
建站之星3.0如何解决常见操作问题?
如何快速配置高效服务器建站软件?
javascript中的try catch异常捕获机制用法分析
*请认真填写需求信息,我们会在24小时内与您取得联系。