传送门

observejs:https://github.com/kmdjs/observejs

本文演示:http://kmdjs.github.io/observejs/list/

本文代码:https://github.com/kmdjs/observejs/tree/master/example/list

写在前面

javascript的编程体验一直被改善,从未停止过。从最早的Jscex(现在的windjs),改善异步编程体验。如今ES6的Generator改善异步编程体验。还有类似的seajs、requirejs提供极致的模块化开发体验;五花八门的Class.js改善面向对象编程体验;kmdjs同时改善模块化编程、面向对象编程和构建体验;各式各样的template.js改善数据->标记的体验。

所有的改善,使代码更直观、友好,使程序易维护、可扩展。

最近使用observejs开发组件,发现有几大优点:

  • Dom操作几乎绝迹
  • 专注于数据操作
  • 视图全自动更新
  • 性能棒棒的
  • 仅一行代码搞定上面四点

本文使用世界上最简单的List组件作为例子,大家通过该例子感受一下observejs组件开发改善之处。

组件代码

  1. var List = function (option) {
  2. this.option = option;
  3. this.data = option.data;
  4. this.parent = document.querySelector(this.option.renderTo);
  5. this.tpl =
  6. '<div class="list-group" style=" text-align: center;width:<%=typeof width === "number"?width+"px":width%>;" >'
  7. + ' <% for ( var i = 0, len = data.length; i < len; i++) { %>'
  8. + '<% var item = data[i]; %>'
  9. + '<a class="list-group-item <%=item.active ? "active" : ""%> <%=item.disabled ? "disabled" : ""%>" href="<%=item.href%>" target="<%=item.target?item.target:""%>"><%=item.text%><\/a>'
  10. + '<% } %>'
  11. + '<\/div>';
  12. this.render();
  13. //list.render建议使用debounce来控制执行频率提高性能,或者和react一样在下次执行requestAnimFrame的时候更新
  14. observe(this, "option", this._debounce(this.render, 200));
  15. }
  16. List.prototype = {
  17. render: function () {
  18. if (this.node) this.parent.removeChild(this.node);
  19. this.parent.innerHTML += this._tpl(this.tpl, this.option);
  20. this.node = this.parent.lastChild;
  21. },
  22. clear:function(){
  23. this.data.size(0);
  24. },
  25. remove:function(index){
  26. this.data.splice(index,1);
  27. },
  28. add:function(item){
  29. this.data.push(item);
  30. },
  31. edit:function(index,item){
  32. this.data[index]=item;
  33. },
  34. disable:function(index){
  35. this.data[index].disabled = true;
  36. },
  37. _tpl: function (str, data) {
  38. var tmpl = 'var __p=[];' + 'with(obj||{}){__p.push(\'' +
  39. str.replace(/\\/g, '\\\\')
  40. .replace(/'/g, '\\\'')
  41. .replace(/<%=([\s\S]+?)%>/g, function (match, code) {
  42. return '\',' + code.replace(/\\'/, '\'') + ',\'';
  43. })
  44. .replace(/<%([\s\S]+?)%>/g, function (match, code) {
  45. return '\');' + code.replace(/\\'/, '\'')
  46. .replace(/[\r\n\t]/g, ' ') + '__p.push(\'';
  47. })
  48. .replace(/\r/g, '\\r')
  49. .replace(/\n/g, '\\n')
  50. .replace(/\t/g, '\\t') +
  51. '\');}return __p.join("");',
  52. func = new Function('obj', tmpl);
  53. return data ? func(data) : func;
  54. },
  55. _debounce: function (func, wait, immediate) {
  56. var timeout;
  57. return function () {
  58. var context = this, args = arguments;
  59. var later = function () {
  60. timeout = null;
  61. if (!immediate) func.apply(context, args);
  62. };
  63. var callNow = immediate && !timeout;
  64. clearTimeout(timeout);
  65. timeout = setTimeout(later, wait);
  66. if (callNow) func.apply(context, args);
  67. };
  68. },
  69. }

ps:模版引擎来自GMU,当然你可以使用任何你喜欢的额模版引擎

组件使用

  1. var list = new List({
  2. data: [
  3. { text: "天下太贰", disabled: true, active: true },
  4. { text: "魔兽争霸", href: "##", target: "_blank" },
  5. { text: "魔兽世界" },
  6. { text: "坦克世界" },
  7. { text: "超级玛丽", disabled: true }
  8. ],
  9. width: 150,
  10. renderTo: "body"
  11. })
  12. list.edit(1,{text:"haha"})
  13. list.add({ text: "aaaa" });
  14. list.remove(2, 1);
  15. list.disable(2);
  16. list.remove(1);
  17. //操作数据等于操作dom
  18. list.data[0].text="aaaaaaa";

组件分析

其中仅需要一行

  1. observe(this, "option", this._debounce(this.render, 100));

就可以实现option的监听,option包含组件的配置,以及数据,数据的变换都能够通知render方法的调用。

可以看到render方法摧毁了整个dom,然后根据数据重新渲染dom。所以控制render的执行频率变得尤其重要。

如果数据变化频繁,render方法调用频繁,从而降低性能,所以,可以看到上面使用debounce控制render执行的频率。

如果不使用debounce,也有第二种方案控制render频率。该方案和react一样,在requestAnimateFrame下次循环的时候去check组件是否要re-render,需要的话就重新渲染。如果一个页面有一百个组件,都统一在requestAnimateFrame循环中check是否re-render。

可以看到,只有render方法里有dom操作。clear、remove、add、edit、disable都只关心数据,数据变了自动会通知到render。

改善的地方在哪里?分两处地方:

1.组件内部代码,内部的增删改查都只关注数据就行

2.组件使用代码,操作数据等同于操作dom

当然如果不使用赋值又没有observejs的情况下可以使用方法调用的方式,使得一行代码什么都能干,但是如果属性太多,你得定义非常多setXXX方法。但你必须知道:赋值胜于method(即【obj.name=xx】>【obj.setName(xx)】)、约定胜于配置:)。

如果你不是很适应react激进虚拟dom和jsx,如果你反感react放弃了HTML+CSS大量优秀特性功能,如果以后dom性能好了,全世界都i78核+ssd了,那么HTML+CSS才是王道啊。

observejs也许是你的另一选择,欢迎尝试,感谢对observejs提出那么多宝贵建议和意见的童鞋。

observejs:https://github.com/kmdjs/observejs

本文演示:http://kmdjs.github.io/observejs/list/

本文代码:https://github.com/kmdjs/observejs/tree/master/example/list

observejs改善组件编程体验的更多相关文章

  1. 基于 OSGi 的面向服务的组件编程,helloworld

    基于 OSGi 的面向服务的组件编程 OSGi(Open Services Gateway Initiative,开放服务网关协议)提供了一个面向服务组件的编程模型,基于 OSGi 编程,具有模块化, ...

  2. 基于 OSGi 的面向服务的组件编程

    作者:曹 羽中 (caoyuz@cn.ibm.com), 软件工程师, IBM中国开发中心 出处:http://www.ibm.com/developerworks/cn/opensource/os- ...

  3. 全新升级的AOP框架Dora.Interception[1]: 编程体验

    多年之前利用IL Emit写了一个名为Dora.Interception(github地址,觉得不错不妨给一颗星)的AOP框架.前几天利用Roslyn的Source Generator对自己为公司写的 ...

  4. kmdjs集成uglifyjs2打造极致的编程体验

    回顾 上篇文章大概展示了kmdjs0.1.x时期的编程范式: 如下面所示,可以直接依赖注入到function里, kmdjs.define('main',['util.bom','app.Ball', ...

  5. Atitit 面向对象编程(OOP)、面向组件编程(COP)、面向方面编程(AOP)和面向服务编程(SOP)的区别和联系

    Atitit 面向对象编程(OOP).面向组件编程(COP).面向方面编程(AOP)和面向服务编程(SOP)的区别和联系 1. 面向组件编程(COP) 所以,组件比起对象来的进步就在于通用的规范的引入 ...

  6. vue.js2.0 自定义组件初体验

    理解 组件(Component)是 Vue.js 最强大的功能之一.组件可以扩展 HTML 元素,封装可重用的代码.在较高层面上,组件是自定义元素, Vue.js 的编译器为它添加特殊功能.在有些情况 ...

  7. DSAPI多功能组件编程应用-HTTP监听服务端与客户端_指令版

    前面介绍了DSAPI多功能组件编程应用-HTTP监听服务端与客户端的内容,这里介绍一个适用于更高效更快速的基于HTTP监听的服务端.客户端. 在本篇,你将见到前所未有的超简化超傻瓜式的HTTP监听服务 ...

  8. Dora.Interception,为.NET Core度身打造的AOP框架 [1]:更加简练的编程体验

    很久之前开发了一个名为Dora.Interception的开源AOP框架(github地址:https://github.com/jiangjinnan/Dora,如果你觉得这个这框架还有那么一点价值 ...

  9. 依赖注入[6]: .NET Core DI框架[编程体验]

    毫不夸张地说,整个ASP.NET Core框架是建立在一个依赖注入框架之上的,它在应用启动时构建请求处理管道过程中,以及利用该管道处理每个请求过程中使用到的服务对象均来源于DI容器.该DI容器不仅为A ...

随机推荐

  1. Hibernate 系列 学习笔记 目录 (持续更新...)

    前言: 最近也在学习Hibernate,遇到的问题差不多都解决了,顺便把学习过程遇到的问题和查找的资料文档都整理了一下分享出来,也算是能帮助更多的朋友们了. 最开始使用的是经典的MyEclipse,后 ...

  2. Linux实战教学笔记05:远程SSH连接服务与基本排错(新手扫盲篇)

    第五节 远程SSH连接服务与基本排错 标签(空格分隔):Linux实战教学笔记-陈思齐 第1章 远程连接LInux系统管理 1.1 为什么要远程连接Linux系统 在实际的工作场景中,虚拟机界面或物理 ...

  3. TCP/IP之Nagle算法与40ms延迟

    Nagle算法是针对网络上存在的微小分组可能会在广域网上造成拥塞而设计的.该算法要求一个TCP连接上最多只能有一个未被确认的未完成的小分组,在该分组确认到达之前不能发送其他的小分组.同时,TCP收集这 ...

  4. HTML 学习笔记 JQuery(DOM 操作2)

    接着上一节的将,这一节从复制节点讲起 复制节点 继续使用之前的例子 如果单击<li>元素后 需要在复制一个<li>元素,可以使用clone()方法完成.全部代码如下 <h ...

  5. 自制文件上传JS控件,支持IE、chrome、firefox etc.

    (function() { if (window.FileUpload) { return; } window.FileUpload = function (id, url) { this.id = ...

  6. Windbg Extension NetExt 使用指南 【3】 ---- 挖掘你想要的数据 Managed Heap

    摘要 : NetExt中有两个比较常用的命令可以用来分析heap上面的对象. 一个是!wheap, 另外一个是!windex. !wheap 这个命令可以用于打印出heap structure信息. ...

  7. 【初探Spring】------Spring IOC(三):初始化过程---Resource定位

    我们知道Spring的IoC起到了一个容器的作用,其中装得都是各种各样的Bean.同时在我们刚刚开始学习Spring的时候都是通过xml文件来定义Bean,Spring会某种方式加载这些xml文件,然 ...

  8. 轻量级表达式树解析框架Faller

    有话说 之前我写了3篇关于表达式树解析的文章 干货!表达式树解析"框架"(1) 干货!表达式树解析"框架"(2) 干货!表达式树解析"框架" ...

  9. Go语言实战 - revel框架教程之CSRF(跨站请求伪造)保护

    CSRF是什么?请看这篇博文“浅谈CSRF攻击方式”,说的非常清楚. 现在做网站敢不防CSRF的我猜只有两种情况,一是没什么人访问,二是局域网应用.山坡网之前属于第一种情况,哈哈,所以至今没什么问题. ...

  10. TODO:MongoDB的查询更新删除总结

    TODO:MongoDB的查询更新删除总结 常用查询,条件操作符查询,< .<=.>.>=.!= 对应 MongoDB的查询操作符是$lt.$lte.$gt.$gte.$ne ...