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 iframe内的函数调用实现方法
Jul 19 Javascript
通过action传过来的值在option获取进行验证的方法
Nov 14 Javascript
node.js入门教程
Jun 01 Javascript
javascript性能优化之DOM交互操作实例分析
Dec 12 Javascript
jQuery获取同级元素的简单代码
Jul 09 Javascript
bootstrap table复杂操作代码
Nov 01 Javascript
如何在AngularJs中调用第三方插件库
May 21 Javascript
微信小程序-getUserInfo回调的实例详解
Oct 27 Javascript
JS实现的找零张数最小问题示例
Nov 28 Javascript
微信小程序实现滚动消息通知
Feb 02 Javascript
vuex + keep-alive实现tab标签页面缓存功能
Oct 17 Javascript
详解vue中v-bind:style效果的自定义指令
Jan 21 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版(2)
2006/10/09 PHP
社区(php&amp;&amp;mysql)五
2006/10/09 PHP
Yii框架安装简明教程
2020/05/15 PHP
Jquery颜色选择器ColorPicker实现代码
2012/11/14 Javascript
open 动态修改img的onclick事件示例代码
2013/11/13 Javascript
jquery如何实现锚点链接之间的平滑滚动
2013/12/02 Javascript
浅谈JavaScript中的string拥有方法的原因
2015/08/28 Javascript
jQuery利用sort对DOM元素进行排序操作
2016/11/07 Javascript
js事件源window.event.srcElement兼容性写法(详解)
2016/11/25 Javascript
jQuery图片轮播功能实例代码
2017/01/29 Javascript
Vue + Webpack + Vue-loader学习教程之相关配置篇
2017/03/14 Javascript
详解vue 模拟后台数据(加载本地json文件)调试
2017/08/25 Javascript
详解使用angular的HttpClient搭配rxjs
2017/09/01 Javascript
vue-cli配置环境变量的方法
2018/07/09 Javascript
浅谈vue限制文本框输入数字的正确姿势
2019/09/02 Javascript
Vue.js数字输入框组件使用方法详解
2019/10/19 Javascript
VUE 实现动态给对象增加属性,并触发视图更新操作示例
2019/11/29 Javascript
js实现移动端吸顶效果
2020/01/08 Javascript
[05:08]顺网杯ISS-DOTA2赛歌 少女偶像Lunar青春演绎
2013/12/05 DOTA
[06:04]DOTA2英雄梦之声Vol19卓尔游侠
2014/06/20 DOTA
python中偏函数partial用法实例分析
2015/07/08 Python
python GUI库图形界面开发之PyQt5动态(可拖动控件大小)布局控件QSplitter详细使用方法与实例
2020/03/06 Python
Python smtp邮件发送模块用法教程
2020/06/15 Python
css3 transform属性详解
2014/09/30 HTML / CSS
利用HTML5实现使用按钮控制背景音乐开关
2015/09/21 HTML / CSS
Agoda西班牙:全球特价酒店预订
2017/06/03 全球购物
英国在线女鞋目的地:SIMMI
2018/12/27 全球购物
Street One瑞士:德国现代时装公司
2019/10/09 全球购物
异常和异常类的概念
2014/09/12 面试题
人事助理自荐信
2014/02/02 职场文书
小学敬老月活动方案
2014/02/11 职场文书
网球场地租赁协议范本
2014/10/07 职场文书
小学教师工作总结2015
2015/04/07 职场文书
公司出差管理制度范本
2015/08/05 职场文书
2016年七夕情人节宣传语
2015/11/25 职场文书
2016暑期校本培训心得体会
2016/01/08 职场文书