JS模板实现方法


Posted in Javascript onApril 03, 2013

概述

我们在使用JS渲染DOM时,一般使用字符串创建DOM然后附加到父元素上,如果附加的DOM是动态易变的,那需要在函数中写大量逻辑。如果在控件实现过程中,这带来的问题更为严重。

解决这个问题的常见解决方案是是使用模板,作为配置项传入控件,实现数据和渲染的分离。具体的实现方法有以下方法:

  1. 字符串替换,使用正则匹配将数据替换进字符串中。
  2. 渲染函数,函数返回字符串。
  3. 模板引擎,可以将执行字符串中的函数(内置或者自定义的)

替换(Substitute)

字符串替换是最简单的实现模板的方式,看一下具体的实现:

1. 定义替换函数

/**
* 替换字符串中的字段.
* @param {String} str 模版字符串
* @param {Object} o json data
* @param {RegExp} [regexp] 匹配字符串的正则表达式
*/
function substitute(str,o,regexp){
return str.replace(regexp || /\\?\{([^{}]+)\}/g, function (match, name) {
return (o[name] === undefined) ? '' : o[name];
});
}

2.使用配置项:

var config = {
data : {value : '123',text:'abc'},
template : '<label>{text}</label><input type="text" value="{value}"/>'
};

3. 在创建DOM的过程中我们这样调用:

var str = substitute(template,data);
$(str).appendTo('body');

通过以上示例,我们就完成了数据和字符串的解耦,可以灵活的用在控件中,当前大多数JS框架都提供了此种方式的模板。

在此基础上可以有下面的扩展,感兴趣的可以自己去实现:

1. 使用数字代替参数名:

如 '<label>{0}</label><input type="text" value="{1}"/>'

2. 嵌套使用对象属性:

如 '<label>{obj.name}</label><input type="text" value="{obj.value}"/>'

优点:实现简单,易于理解。

缺点:只能进行简单的数据结构,无法处理循环、条件语句。

渲染方法(Render)

我们可以在渲染函数中处理非常复杂的逻辑,可以将渲染函数作为参数传入配置项。

配置项:

var config = {
data : [{value : '0',text:'abc'},{value : '1',text:'bcd'}],
renderer : function(obj){
if(obj.value === '0'){
return obj.text;
}else{
return '<img title="' + obj.text + '" src=""/>';
}
}
};

在使用时:

for(var i = 0 ; i< data.length; i++){
var obj = data[i],
str = config.renderer(obj);
$(str).appendTo('body');
}

在处理循环,条件语句时,这是一种很好的解决方案。

优点:实现相对简单,实现灵活,能满足复杂数据结构,易于调试

缺点:

  1. 渲染函数作为配置项,不易理解。
  2. 函数较长时,使配置项臃肿。
  3. 每个场景都需要自己实现渲染函数。

模板引擎(XTemplate)

每一个JS UI库都会有一个功能强大的模板引擎,一个模板引擎需要实现以下功能:

1. 字符串替换

2. 处理复杂语句 条件、循环

3. 使用内嵌函数

4. 允许用户传入自定义函数

目前的模板引擎有2种常见的实现方式:

1. 使用正则分析字符串,执行其中的特殊语句逻辑,替换对应的数据

我们来看一下KISSY 模板的一个实例:

'Hello, {{#each users}}{{#if _ks_value.show}}{{_ks_value.name}}{{/if}}{{/each}}.'

上面这是一个模板,可以处理循环、条件语句。

2. 对字符串进行语法分析,生成语法树,执行替换对应的标签或数据。

下面是Ext的 xtemplate使用方式:

var tpl = new Ext.XTemplate(
'<p>{name}\'s favorite beverages:</p>',
'<tpl for="drinks">',
'<div> - {.}</div>',
'</tpl>'
);
tpl.overwrite(panel.body, data);

优点:功能强大,灵活性高

缺点:使用复杂,更加不易理解。不便于调试。

问题思考

1. 控件中使用模板,可以将数据和DOM分离,但是如果一个控件中包含大量的模板,会增加使用者的工作量,而且不易于调试,需要权衡使用。

2. 如果大量控件使用相同的模板,和相同的数据结构,每个控件单独配置不便于使用,更好的方案是允许父控件配置模板。

 

Javascript 相关文章推荐
JavaScript 实现??打印?理
Apr 28 Javascript
javascript下判断一个对象是否具有指定名称的属性的的代码
Jan 11 Javascript
Jquery attr(&quot;checked&quot;) 返回checked或undefined 获取选中失效
Oct 10 Javascript
Javascript的setTimeout()使用闭包特性时需要注意的问题
Sep 23 Javascript
实现easyui的datagrid导出为excel的示例代码
Nov 10 Javascript
快速将Vue项目升级到webpack3的方法步骤
Sep 14 Javascript
关于react中组件通信的几种方式详解
Dec 10 Javascript
Angular4集成ng2-file-upload的上传组件
Mar 14 Javascript
讲解vue-router之什么是编程式路由
May 28 Javascript
解决jquery的ajax调取后端数据成功却渲染失败的问题
Aug 08 jQuery
基于Vue 撸一个指令实现拖拽功能
Oct 09 Javascript
JavaScript 声明私有变量的两种方式
Feb 05 Javascript
JS实现标签页效果(配合css)
Apr 03 #Javascript
JS实现下拉框的动态添加(附效果)
Apr 03 #Javascript
js去字符串前后空格5种实现方法及比较
Apr 03 #Javascript
终于解决了IE8不支持数组的indexOf方法
Apr 03 #Javascript
jQuery布局插件UI Layout简介及使用方法
Apr 03 #Javascript
简单实例处理url特殊符号&amp;处理(2种方法)
Apr 02 #Javascript
JS获取浏览器版本及名称实现函数
Apr 02 #Javascript
You might like
完美实现GIF动画缩略图的php代码
2011/01/02 PHP
在smarty模板中使用PHP函数的方法
2011/04/23 PHP
PHP中数组的三种排序方法分享
2012/05/07 PHP
简单的php新闻发布系统教程
2014/05/09 PHP
十个PHP高级应用技巧果断收藏
2015/09/25 PHP
基于PHP实现简单的随机抽奖小程序
2016/01/05 PHP
Smarty模板类内部原理实例分析
2019/07/03 PHP
Javascript alert消息换行的方法
2013/08/07 Javascript
快速解决jquery之get缓存问题的最简单方法介绍
2013/12/19 Javascript
使用DNode实现php和nodejs之间通信的简单实例
2015/07/06 NodeJs
解决bootstrap导航栏navbar在IE8上存在缺陷的方法
2016/07/01 Javascript
Javascript类型系统之undefined和null浅析
2016/07/13 Javascript
jquery计算出left和top,让一个div水平垂直居中的简单实例
2016/07/13 Javascript
JavaScript随机打乱数组顺序之随机洗牌算法
2016/08/02 Javascript
在javascript中使用com组件的简单实现方法
2016/08/17 Javascript
AngularJS $injector 依赖注入详解
2016/09/14 Javascript
微信小程序教程系列之新建页面(4)
2017/04/17 Javascript
Javascript调试之console对象——你不知道的一些小技巧
2017/07/10 Javascript
基于 Vue.js 2.0 酷炫自适应背景视频登录页面实现方式
2018/01/17 Javascript
nvm、nrm、npm 安装和使用详解(小结)
2019/01/17 Javascript
js的继承方法小结(prototype、call、apply)(推荐)
2019/04/17 Javascript
JavaScript中的连续赋值问题实例分析
2019/07/12 Javascript
element中的$confirm的使用
2020/04/26 Javascript
通过实例解析json与jsonp原理及使用方法
2020/09/27 Javascript
Python在不同目录下导入模块的实现方法
2017/10/27 Python
python生成tensorflow输入输出的图像格式的方法
2018/02/12 Python
详解如何在Apache中运行Python WSGI应用
2019/01/02 Python
PyQt5实现简单数据标注工具
2019/03/18 Python
python实现AES和RSA加解密的方法
2019/03/28 Python
英国性感内衣和睡衣品牌:Bluebella
2018/01/26 全球购物
size?瑞典:英国伦敦的球鞋精品店
2018/03/01 全球购物
奥地利手表、香水、化妆品和珠宝购物网站:Brasty.at
2021/01/17 全球购物
优秀管理者事迹材料
2014/05/22 职场文书
2014年度工作总结报告
2014/12/15 职场文书
2014年高中教师工作总结
2014/12/19 职场文书
Java网络编程之UDP实现原理解析
2021/09/04 Java/Android