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打开新窗口同时关闭旧窗口
Jan 16 Javascript
Jquery实现图片左右自动滚动示例
Sep 25 Javascript
多种方法实现load加载完成后把图片一次性显示出来
Feb 19 Javascript
jQuery EasyUI提交表单验证
Jul 19 Javascript
基于JavaScript实现点击页面任何位置返回
Aug 31 Javascript
Three.js的使用及绘制基础3D图形详解
Apr 27 Javascript
微信小程序多张图片上传功能
Jun 07 Javascript
基于JS对象创建常用方式及原理分析
Jun 28 Javascript
js处理包含中文的字符串实例
Oct 11 Javascript
Vue中this.$router.push参数获取方法
Feb 27 Javascript
JS中获取 DOM 元素的绝对位置实例详解
Apr 23 Javascript
微信公众号获取用户地理位置并列出附近的门店的示例代码
Jul 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中的类-什么叫类
2006/11/20 PHP
php准确计算复活节日期的方法
2015/04/18 PHP
基于ThinkPHP5.0实现图片上传插件
2017/09/25 PHP
laravel框架上传图片实现实时预览功能
2019/10/14 PHP
动态控制Table的js代码
2007/03/07 Javascript
JavaScript.The.Good.Parts阅读笔记(二)作用域&amp;闭包&amp;减缓全局空间污染
2010/11/16 Javascript
基于jquery的从一个页面跳转到另一个页面的指定位置的实现代码(带平滑移动的效果)
2011/05/24 Javascript
jQuery获取对象简单实现方法小结
2014/10/30 Javascript
JavaScript跨浏览器获取页面中相同class节点的方法
2015/03/03 Javascript
javascript+html5实现仿flash滚动播放图片的方法
2015/04/27 Javascript
全面解析jQuery $(document).ready()和JavaScript onload事件
2016/06/08 Javascript
jquery判断页面网址是否有效的两种方法
2016/12/11 Javascript
vue-cli项目中怎么使用mock数据
2017/09/27 Javascript
解决vue 更改计算属性后select选中值不更改的问题
2018/03/02 Javascript
Vue实现6位数密码效果
2018/08/18 Javascript
微信小程序picker组件关于objectArray数据类型的绑定方法
2019/03/13 Javascript
使用Node.js实现一个多人游戏服务器引擎
2019/03/13 Javascript
解决layer弹出层自适应页面大小的问题
2019/09/16 Javascript
浅谈Python 中整型对象的存储问题
2016/05/16 Python
python基础教程项目五之虚拟茶话会
2018/04/02 Python
实例讲解Python中浮点型的基本内容
2019/02/11 Python
浅谈Tensorflow 动态双向RNN的输出问题
2020/01/20 Python
Python如何使用bokeh包和geojson数据绘制地图
2020/03/21 Python
python读取配置文件方式(ini、yaml、xml)
2020/04/09 Python
使用python实现名片管理系统
2020/06/18 Python
pytorch实现查看当前学习率
2020/06/24 Python
Python数据分析库pandas高级接口dt的使用详解
2020/12/11 Python
野兽派官方旗舰店:THE BEAST 野兽派
2016/08/05 全球购物
La Redoute英国官网:法国时尚品牌
2017/04/27 全球购物
需求分析说明书
2014/05/09 职场文书
2014优秀党员事迹材料
2014/08/14 职场文书
小学生感恩父母演讲稿
2014/08/28 职场文书
行政申诉状范文
2015/05/20 职场文书
小学音乐课教学反思
2016/02/18 职场文书
分析MySQL抛出异常的几种常见解决方式
2021/05/18 MySQL
分享五个Node.js开发的优秀实践 
2022/04/07 NodeJs