JavaScript 模板引擎实现原理解析生活记载

本文介绍了JavaScript模板引擎的工作原理,通过实例展示了如何将数据导入到模板中,包括简单的字符串替换和正则表达式替换。同时,探讨了模板存放、获取、函数、解析编译和分隔符等相关知识,并对jQuery tmpl的实现进行了简要解析。
摘要由CSDN通过智能技术生成

首先我们来看一个简单模板:

复制代码
复制代码

复制代码
复制代码
其中被{{ xxx }}包含的就是我们要替换的变量。
接着我们可能通过ajax或者其他方法获得数据。这里我们自己定义了数据,具体如下:

复制代码
复制代码
var data = [
{
title: “Create a Sticky Note Effect in 5 Easy Steps with CSS3 and HTML5”,
href: “http://net.tutsplus.com/tutorials/html-css-techniques/create-a-sticky-note-effect-in-5-easy-steps-with-css3-and-html5/”,
imgSrc: “https://d2o0t5hpnwv4c1.cloudfront.net/771_sticky/sticky_notes.jpg”
},
{
title: “Nettuts+ Quiz #8”,
href: “http://net.tutsplus.com/articles/quizzes/nettuts-quiz-8-abbreviations-darth-sidious-edition/”,
imgSrc: “https://d2o0t5hpnwv4c1.cloudfront.net/989_quiz2jquerybasics/quiz.jpg”
}
];
复制代码
复制代码
ok,现在的问题就是我们怎么把数据导入到模板里面呢?

第一种大家会想到的就是采用replace直接替换里面的变量:

复制代码
复制代码
template = document.querySelector(’#template’).innerHTML,
result = document.querySelector(’.result’),
i = 0, len = data.length,
fragment = ‘’;

for ( ; i < len; i++ ) {
fragment += template
.replace( /{{title}}/, data[i].title )
.replace( /{{href}}/, data[i].href )
.replace( /{{imgSrc}}/, data[i].imgSrc );
}

result.innerHTML = fragment;
复制代码
复制代码
第二种的话,相对第一种比较灵活,采用的是正则替换,对于初级前端,很多人对正则掌握的并不是很好,一般也用的比较少。具体实现如下:

复制代码
复制代码
template = document.querySelector(’#template’).innerHTML,
result = document.querySelector(’.result’),
attachTemplateToData;

// 将模板和数据作为参数,通过数据里所有的项将值替换到模板的标签上(注意不是遍历模板标签,因为标签可能不在数据里存在)。
attachTemplateToData = function(template, data) {
var i = 0,
len = data.length,
fragment = ‘’;

    // 遍历数据集合里的每一个项,做相应的替换
    function replace(obj) {
        var t, key, reg;

//遍历该数据项下所有的属性,将该属性作为key值来查找标签,然后替换
for (key in obj) {
reg = new RegExp(’{{’ + key + ‘}}’, ‘ig’);
t = (t || template).replace(reg, obj[key]);
}

        return t;
    }

    for (; i < len; i++) {
        fragment += replace(data[i]);
    }

    return fragment;
};

result.innerHTML = attachTemplateToData(template, data);
复制代码
复制代码
与第一种相比较,第二种代码看上去多了,但是功能实则更为强大了。第一种我们需要每次重新编写变量名,如果变量名比较多的话,会比较麻烦,且容易出错。第二种的就没有这些烦恼。

2、模板引擎相关知识
通过上面的例子,大家对模板引擎应该有个初步的认识了,下面我们来讲解一些相关知识。

2.1 模板存放
模板一般都是放置到 textarea/input 等表单控件,或者 script 等标签中。比如上面的例子,我们就是放在 script 标签上的。

2.2 模板获取
一般都是通过ID来获取,document.getElementById(“ID”):

//textarea或input则取value,其它情况取innerHTML
var html = /^(textarea|input)$/i.test(element.nodeName) ? element.value : element.innerHTML;
上面的是通用的模板获取方法,这样不管你是放在 textarea/input 还是 script 标签下都可以获取到。

2.3 模板函数
一般都是templateFun(“id”, data);其中id为存放模板字符串的元素id,data为需要装载的数据。

2.4 模板解析编译
模板解析主要是指将模板中 JavaScript 语句和 html 分离出来,编译的话将模板字符串编译成最终的模板。上面的例子比较简单,还没有涉及到模板引擎的核心。

2.5 模板分隔符
要指出的是,不同的模板引擎所用的分隔符可能是不一样,上面的例子用的是{{ }},而Jquery tmpl 使用的是<% %>。

3、jQuery tmpl 实现原理解析
jQuery tmpl是由jQuery的作者写的,代码短小精悍。总共20多行,功能却比我们上面的强大很多。我们先来看一看源码:

复制代码
复制代码
(function(){
var cache = {};

this.tmpl = function tmpl(str, data){

var fn = !/\W/.test(str) ? 
  cache[str] = cache[str] ||
    tmpl(document.getElementById(str).innerHTML) :

  new Function("obj",
    "var p=[],print=function(){p.push.apply(p,arguments);};" +
   
    "with(obj){p.push('" +
   
    str
      .replace(/[\r\t\n]/g, " ") 
      .split("<%").join("\t") 
      .replace(/((^|%>)[^\t]*)'/g, "$1\r")
      .replace(/\t=(.*?)%>/g, "',$1,'")  
      .split("\t").join("');")  
      .split("%>").join("p.push('") 
      .split("\r").join("\\'")
  + "');}return p.join('');");

return data ? fn( data ) : fn;

};
})();
复制代码
复制代码
初看是不是觉得有点懵,完全不能理解的代码。没事,后面我们会对源码进行解释的,我们还是先看一下所用的模板

可以发现,这个模板比入门例子的模板更为复杂,因为里面还夹杂着 JavaScript 代码。JavaScript 代码采用 <% %> 包含。而要替换的变量则是用 <%= %> 分隔开的。

下面我再来对代码做个注释。不过即使看了注释,你也不一定能很快理解,最好的办法是自己实际动手操作一遍。

复制代码
复制代码
// 代码整个放在一个立即执行函数里面
(function(){
// 用来缓存,有时候一个模板要用多次,这时候,我们直接用缓存就会很方便
var cache = {};

// tmpl绑定在this上,这里的this值得是window
this.tmpl = function tmpl(str, data){

// 只有模板才有非字母数字字符,用来判断传入的是模板id还是模板字符串,
// 如果是id的话,判断是否有缓存,没有缓存的话调用tmpl;
// 如果是模板的话,就调用new Function()解析编译
var fn = !/\W/.test(str) ? 
  cache[str] = cache[str] ||
    tmpl(document.getElementById(str).innerHTML) :

  new Function("obj",

// 注意这里整个是字符串,通过 + 号拼接
“var p=[],print=function(){p.push.apply(p,arguments);};” +
“with(obj){p.push(’” +

    str

// 去除换行制表符\t\n\r
.replace(/[\r\t\n]/g, " “)
      
      // 将左分隔符变成 \t
.split(”<%").join("\t")
      
      // 去掉模板中单引号的干扰
.replace(/((|%>)[\t])’/g, “$1\r”)
      
      // 为 html 中的变量变成 “,xxx,” 的形式, 如:\t=users[i].url%> 变成 ‘,users[i].url,’
      // 注意这里只有一个单引号,还不配对
.replace(/\t=(.
?)%>/g, “’,$1,’”)
      
      // 这时候,只有JavaScript 语句前面才有 “\t”, 将 \t 变成 ‘);
      // 这样就可把 html 标签添加到数组p中,而javascript 语句 不需要 push 到里面。
      .split("\t").join("’);")
      
      // 这时候,只有JavaScript 语句后面才有 “%>”, 将 %> 变成 p.push(’
      // 上一步我们再 html 标签后加了 ‘);, 所以要把 p.push(’ 语句放在 html 标签放在前面,这样就可以变成 JavaScript 语句
.split("%>").join(“p.push(’”)

// 将上面可能出现的干扰的单引号进行转义
      .split("\r").join("\’")
    // 将数组 p 变成字符串。
+ “’);}return p.join(’’);”);

return data ? fn( data ) : fn;

};
})();
复制代码
复制代码
上面代码中,有一个要指出的就是new Function 的使用 方法。给 new Function() 传一个字符串作为函数的body来构造一个 JavaScript函数。编程中并不经常用到,但有时候应该是很有用的。

下面是 new Function 的基本用法:

// 最后一个参数是函数的 body(函数体),类型为 string;
// 前面的参数都是 索要构造的函数的参数(名字)
var myFunction = new Function(‘users’, ‘salary’, ‘return users * salary’);
最后的字符串就是下面这种形式:

复制代码
复制代码
var p = [],
print = function() {
p.push.apply(p, arguments);
};
with(obj) {
p.push(’

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值