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 相关文章推荐
jQuery下扩展插件和拓展函数的写法(匿名函数使用的典型例子)
Oct 20 Javascript
Javascript insertAfter() 实现函数代码
Oct 12 Javascript
使用CSS3的scale实现网页整体缩放
Mar 18 Javascript
javascript显式类型转换实例分析
Apr 25 Javascript
jQuery操作Table技巧大汇总
Jan 23 Javascript
jQuery的Cookie封装,与PHP交互的简单实现
Oct 05 Javascript
JS常用加密编码与算法实例总结
Dec 22 Javascript
Cropper.js 实现裁剪图片并上传(PC端)
Aug 20 Javascript
浅谈实现vue2.0响应式的基本思路
Feb 13 Javascript
vue iview实现动态路由和权限验证功能
Apr 17 Javascript
利用hasOwnProperty给数组去重的面试题分享
Nov 05 Javascript
js实现简单的日历显示效果函数示例
Nov 25 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
PHP Zip压缩 在线对文件进行压缩的函数
2010/05/26 PHP
PHP根据IP地址获取所在城市具体实现
2013/11/27 PHP
在PHP中使用X-SendFile头让文件下载更快
2014/06/01 PHP
PHP简单实现生成txt文件到指定目录的方法
2016/04/25 PHP
php使用高斯算法实现图片的模糊处理功能示例
2016/11/11 PHP
PHP简单实现合并2个数字键数组值的方法
2017/05/30 PHP
PHP实现websocket通信的方法示例
2018/08/28 PHP
JavaScript语法着色引擎(demo及打包文件下载)
2007/06/13 Javascript
一句话JavaScript表单验证代码
2009/08/02 Javascript
JQuery选中checkbox方法代码实例(全选、反选、全不选)
2015/04/27 Javascript
JavaScript中判断函数、变量是否存在
2015/06/10 Javascript
js焦点文字滚动效果代码分享
2015/08/25 Javascript
一波JavaScript日期判断脚本分享
2016/03/06 Javascript
jQuery图片渐变特效的简单实现
2016/06/25 Javascript
JavaScript实现简易的天数计算器实例【附demo源码下载】
2017/01/18 Javascript
JQuery.dataTables表格插件添加跳转到指定页
2017/06/09 jQuery
微信小程序 空白页重定向解决办法
2017/06/27 Javascript
JavaScript实现三级级联特效
2017/11/05 Javascript
JS装饰器函数用法总结
2018/04/21 Javascript
详解angularjs4部署文件过大解决过程
2018/12/05 Javascript
详解Vue+ElementUI从零开始搭建自己的网站(一、环境搭建)
2019/04/30 Javascript
jquery实现自定义树形表格的方法【自定义树形结构table】
2019/07/12 jQuery
highcharts.js数据绑定方式代码实例
2019/11/13 Javascript
javascript实现随机抽奖功能
2020/12/30 Javascript
基于vuex实现购物车功能
2021/01/10 Vue.js
python中stdout输出不缓存的设置方法
2014/05/29 Python
python开发之thread线程基础实例入门
2015/11/11 Python
教你用一行Python代码实现并行任务(附代码)
2018/02/02 Python
使用Python设计一个代码统计工具
2018/04/04 Python
使用Python制作一盏 3D 花灯喜迎元宵佳节
2021/02/26 Python
canvas需要在标签里直接定义宽高
2014/12/17 HTML / CSS
雪花秀美国官方网站:韩国著名草本护肤化妆品品牌
2016/10/19 全球购物
Nike挪威官网:Nike.com (NO)
2018/11/26 全球购物
班组拓展活动方案
2014/08/14 职场文书
干货!开幕词的写作方法
2019/04/02 职场文书
vue引入Excel表格插件的方法
2021/04/28 Vue.js