underscore 提供了模板引擎的功能,举个例子:javascript
var tpl = "hello: <%= name %>";
var compiled = _.template(tpl);
compiled({name: 'Kevin'}); // "hello: Kevin"
复制代码
感受好像没有什么强大的地方,再来举个例子:html
在 HTML 文件中:java
<ul id="name_list"></ul>
<script type="text/html" id="user_tmpl"> <%for ( var i = 0; i < users.length; i++ ) { %> <li> <a href="<%=users[i].url%>"> <%=users[i].name%> </a> </li> <% } %> </script>
复制代码
JavaScript 文件中:git
var container = document.getElementById("user_tmpl");
var data = {
users: [
{ "name": "Kevin", "url": "http://localhost" },
{ "name": "Daisy", "url": "http://localhost" },
{ "name": "Kelly", "url": "http://localhost" }
]
}
var precompile = _.template(document.getElementById("user_tmpl").innerHTML);
var html = precompile(data);
container.innerHTML = html;
复制代码
效果为:github
那么该如何实现这样一个 _.template 函数呢?bash
underscore 的 template 函数参考了 jQuery 的做者 John Resig 在 2008 年发表的一篇文章 JavaScript Micro-Templating,咱们先从这篇文章的思路出发,思考一下如何写一个简单的模板引擎。数据结构
依然是以这段模板字符串为例:闭包
<%for ( var i = 0; i < users.length; i++ ) { %>
<li> <a href="<%=users[i].url%>"> <%=users[i].name%> </a> </li>
<% } %>
复制代码
John Resig 的思路是将这段代码转换为这样一段程序:架构
// 模拟数据
var users = [{"name": "Kevin", "url": "http://localhost"}];
var p = [];
for (var i = 0; i < users.length; i++) {
p.push('<li><a href="');
p.push(users[i].url);
p.push('">');
p.push(users[i].name);
p.push('</a></li>');
}
// 最后 join 一下就能够获得最终拼接好的模板字符串
console.log(p.join('')) // <li><a href="http://localhost">Kevin</a></li>
复制代码
咱们注意,模板实际上是一段字符串,咱们怎么根据一段字符串生成一段代码呢?很容易就想到用 eval,那咱们就先用 eval 吧。函数
而后咱们会发现,为了转换成这样一段代码,咱们须要将<%xxx%>
转换为 xxx
,其实就是去掉包裹的符号,还要将 <%=xxx%>
转化成 p.push(xxx)
,这些均可以用正则实现,可是咱们还须要写 p.push('<li><a href="');
、p.push('">');
呐,这些该如何实现呢?
那咱们换个思路,依然是用正则,可是咱们
%>
替换成 p.push('
<%
替换成 ');
<%=xxx%>
替换成 ');p.push(xxx);p.push('
咱们来举个例子:
<%for ( var i = 0; i < users.length; i++ ) { %>
<li>
<a href="<%=users[i].url%>">
<%=users[i].name%>
</a>
</li>
<% } %>
复制代码
按照这个替换规则会被替换为:
');for ( var i = 0; i < users.length; i++ ) { p.push('
<li>
<a href="');p.push(users[i].url);p.push('">
');p.push(users[i].name);p.push('
</a>
</li>
'); } p.push('
复制代码
这样确定会报错,毕竟代码都没有写全,咱们在首和尾加上部分代码,变成:
// 添加的首部代码
var p = []; p.push(' ');for ( var i = 0; i < users.length; i++ ) { p.push(' <li> <a href="');p.push(users[i].url);p.push('"> ');p.push(users[i].name);p.push(' </a> </li> '); } p.push(' // 添加的尾部代码 ');
复制代码
咱们整理下这段代码:
var p = []; p.push('');
for ( var i = 0; i < users.length; i++ ) {
p.push('<li><a href="');
p.push(users[i].url);
p.push('">');
p.push(users[i].name);
p.push('</a></li>');
}
p.push('');
复制代码
刚好能够实现这个功能,不过还要注意一点,要将换行符替换成空格,防止解析成代码的时候报错,不过在这里为了方便理解原理,就只在代码里实现。
咱们来尝试实现初版:
// 初版
function tmpl(str, data) {
var str = document.getElementById(str).innerHTML;
var string = "var p = []; p.push('" +
str
.replace(/[\r\t\n]/g, "")
.replace(/<%=(.*?)%>/g, "');p.push($1);p.push('")
.replace(/<%/g, "');")
.replace(/%>/g,"p.push('")
+ "');"
eval(string)
return p.join('');
};
复制代码
为了验证是否有用:
HTML 文件:
<script type="text/html" id="user_tmpl"> <%for ( var i = 0; i < users.length; i++ ) { %> <li> <a href="<%=users[i].url%>"> <%=users[i].name%> </a> </li> <% } %> </script>
复制代码
JavaScript 文件:
var users = [
{ "name": "Byron", "url": "http://localhost" },
{ "name": "Casper", "url": "http://localhost" },
{ "name": "Frank", "url": "http://localhost" }
]
tmpl("user_tmpl", users)
复制代码
完整的 Demo 能够查看 template 示例一
在这里咱们使用了 eval ,实际上 John Resig 在文章中使用的是 Function 构造函数。
Function 构造函数建立一个新的 Function 对象。 在 JavaScript 中, 每一个函数实际上都是一个 Function 对象。
使用方法为:
new Function ([arg1[, arg2[, ...argN]],] functionBody)
复制代码
arg1, arg2, ... argN 表示函数用到的参数,functionBody 表示一个含有包括函数定义的 JavaScript 语句的字符串。
举个例子:
var adder = new Function("a", "b", "return a + b");
adder(2, 6); // 8
复制代码
那么 John Resig 究竟是如何实现的呢?
使用 Function 构造函数:
// 第二版
function tmpl(str, data) {
var str = document.getElementById(str).innerHTML;
var fn = new Function("obj",
"var p = []; p.push('" +
str
.replace(/[\r\t\n]/g, "")
.replace(/<%=(.*?)%>/g, "');p.push($1);p.push('")
.replace(/<%/g, "');")
.replace(/%>/g,"p.push('")
+ "');return p.join('');");
return fn(data);
};
复制代码
使用方法依然跟初版相同,具体 Demo 能够查看 template 示例二
不过值得注意的是:其实 tmpl 函数没有必要传入 data 参数,也没有必要在最后 return 的时候,传入 data 参数,即便你把这两个参数都去掉,代码仍是能够正常执行的。
这是由于:
使用Function构造器生成的函数,并不会在建立它们的上下文中建立闭包;它们通常在全局做用域中被建立。当运行这些函数的时候,它们只能访问本身的本地变量和全局变量,不能访问Function构造器被调用生成的上下文的做用域。这和使用带有函数表达式代码的 eval 不一样。
这里之因此依然传入了 data 参数,是为了下一版作准备。
如今有一个小问题,就是实际上咱们传入的数据结构可能比较复杂,好比:
var data = {
status: 200,
name: 'kevin',
friends: [...]
}
复制代码
若是咱们将这个数据结构传入 tmpl 函数中,在模板字符串中,若是要用到某个数据,老是须要使用 data.name
、data.friends
的形式来获取,麻烦就麻烦在我想直接使用 name、friends 等变量,而不是繁琐的使用 data.
来获取。
这又该如何实现的呢?答案是 with。
with 语句能够扩展一个语句的做用域链(scope chain)。当须要屡次访问一个对象的时候,可使用 with 作简化。好比:
var hostName = location.hostname;
var url = location.href;
// 使用 with
with(location){
var hostname = hostname;
var url = href;
}
复制代码
function Person(){
this.name = 'Kevin';
this.age = '18';
}
var person = new Person();
with(person) {
console.log('my name is ' + name + ', age is ' + age + '.')
}
// my name is Kevin, age is 18.
复制代码
最后:不建议使用 with 语句,由于它多是混淆错误和兼容性问题的根源,除此以外,也会形成性能低下
使用 with ,咱们再写一版代码:
// 第三版
function tmpl(str, data) {
var str = document.getElementById(str).innerHTML;
var fn = new Function("obj",
// 其实就是这里多添加了一句 with(obj){...}
"var p = []; with(obj){p.push('" +
str
.replace(/[\r\t\n]/g, "")
.replace(/<%=(.*?)%>/g, "');p.push($1);p.push('")
.replace(/<%/g, "');")
.replace(/%>/g,"p.push('")
+ "');}return p.join('');");
return fn(data);
};
复制代码
具体 Demo 能够查看 template 示例三
若是咱们的模板不变,数据却发生了变化,若是使用咱们的以前写的 tmpl 函数,每次都会 new Function,这实际上是没有必要的,若是咱们能在使用 tmpl 的时候,返回一个函数,而后使用该函数,传入不一样的数据,只根据数据不一样渲染不一样的 html 字符串,就能够避免这种无谓的损失。
// 第四版
function tmpl(str, data) {
var str = document.getElementById(str).innerHTML;
var fn = new Function("obj",
"var p = []; with(obj){p.push('" +
str
.replace(/[\r\t\n]/g, "")
.replace(/<%=(.*?)%>/g, "');p.push($1);p.push('")
.replace(/<%/g, "');")
.replace(/%>/g,"p.push('")
+ "');}return p.join('');");
var template = function(data) {
return fn.call(this, data)
}
return template;
};
// 使用时
var compiled = tmpl("user_tmpl");
results.innerHTML = compiled(data);
复制代码
具体 Demo 能够查看 template 示例四
至此,咱们已经跟着 jQuery 的做者 John Resig 实现了一个简单的模板引擎,虽然 underscore 基于这个思路实现,可是功能强大,相对的,代码也更加复杂一下,下一篇,咱们一块儿去分析 underscore 的 template 函数实现。
underscore 系列目录地址:github.com/mqyqingfeng…。
underscore 系列预计写八篇左右,重点介绍 underscore 中的代码架构、链式调用、内部函数、模板引擎等内容,旨在帮助你们阅读源码,以及写出本身的 undercore。
若是有错误或者不严谨的地方,请务必给予指正,十分感谢。若是喜欢或者有所启发,欢迎 star,对做者也是一种鼓励。