由数据绑定和排序引入的几个JavaScript知识点

一、sort深入

这次我们要完成一个表格排序的案例,那说到排序,最终肯定是归结于数组的排序。对于数组的排序,我们最先想到的肯定是sort方法,所以我们深入了解一下sort。

1.1、基本排序

定义数组:

var ary = [22, 3, 14, 12, 23, 1, 48];

在mdn中对sort的定义如下:

arr.sort(compareFunction)
compareFunction:可选。用来指定按某种顺序进行排列的函数。如果省略,元素按照转换为的字符串的诸个字符的Unicode位点进行排序。

从mdn的描述中可以看出,如果没有指定一个函数作为参数,那么只能比较10以内的数字,所以一般都会指定一个函数作为参数。

在compareFunction中可以传入两个参数假定为a、b,这两个参数即为待比较多两个数:

  • a:每一次执行匿名函数的时候,找到的数组中的当前项;
  • b:当前项的后一项

compareFunction的返回值则为一个数字,如果返回值大于0,则让a和b交换一下位置,如果返回值小于等于0,原来的位置不动。

在本例中:

  • 如果:return a - b,表示升序排序,如果a大于b,返回值大于0,a和b交换位置,输出的结果为:[1, 3, 12, 14, 22, 23, 48]

  • 如果:return b - a,表示降序排序,如果b大于a,返回值大于0,a和b交换位置,输出的结果为:[48, 23, 22, 14, 12, 3, 1]

1.2、对象数组(二维数组)排序

定义对象数组:

var ary = [
    {name: 'iceman', age: 25},
    {name: 'mengzhe', age: 13},
    {name: 'shoushou', age: 107},
    {name: 'jiajia', age: 256}
];

对象数组的排序,正常情况都是要求根据某个值为数字的属性来排序,在本例中,我们要求以年龄来排序。

sort方法的参数compareFunction的参数中两个参数,并一定就一定是要数字,这两个参数是数组中的某一项,注意是数组中的某一项哦!也就是说在对象数组中,这两个参数代表的是一个对象。在得到了这个结论之后,我们就可以根据通过对象的属性来排序了:

ary.sort(function (a, b) {
    return parseFloat(a.age) - parseFloat(b.age);
});
console.log(ary);
对象数组排序后的结果.png

二、json

现在与服务端的交互中一般都是使用json,应该基本被使用xml了,我是做Android开发的,在我开始做Android的时候就是使用json,所以json也是现在主流的一种数据交互格式。。

可能很多人会将json归结于一种新的数据类型,其实并不是这样的,json只是一种特殊的数据格式,归根结底json还是对象数据类型的。比如,

普通格式的对象如下:

var obj = {name:'iceman' , age:7};

json格式对象如下:

var jsonObj = {"name":"iceman" , "age":7};

相对于普通格式的对象来说,json对象只是把属性名用双引号包起来了。

在window浏览器对象中,提供了一个叫做JSON的属性(window.JSON),它里面提供了两个方法:

  • JSON.parse :把JSON格式的字符串,转换为JSON格式的对象;
  • JSON.stringify :把JSON格式的对象,转换为JSON格式的字符串;
var jsonObj = {"name":"iceman" , "age":7};
var jsonStr = JSON.stringify(jsonObj); // --> 字符串
console.log(jsonStr);

var str = '{"name":"iceman" , "age":7}';
console.log(JSON.parse(str));

注意: 在IE6~7中,window下没有JSON属性,所以parse和stringify方法都不存在了,那么这时候要把JSON格式的字符串转换为JSON格式的对象可以使用eval方法:

var str = '{"name":"iceman" , "age":7}';
eval("(" + str + ")");

使用eval装好JSON格式的对象,一定要手动加一个小括号。

二、数据绑定

2.1、准备JSON格式的数据

var ary = [
    {
        "title": "11111",
        "desc": "aaaaaaaa"
    },
    {
        "title": "22222",
        "desc": "bbbbbbbb"
    }
  ......
];

2.2、页面结构

<ul id="ul1">
    <li>原来就有的,这里有鼠标的移入移出事件</li>
</ul>
body, ul, li {margin: 0; padding: 0; list-style: none;}
#ul1 {margin: 10px auto; padding: 10px; width: 300px; border:1px solid #008000;  }
#ul1 li {position: relative; padding-left: 28px;  height:35px; line-height: 35px; text-overflow: ellipsis; white-space: nowrap; overflow: hidden;  }
#ul1 li span { display: block; position: absolute;  top: 6px;  left: 0;  width: 21px;  height: 21px;  line-height: 20px;  text-align: center;  border: 1px solid #ccc;  font-size: 12px;  border-radius: 50%;  }

寻找元素以及添加移入移出事件:

var oUl = document.getElementById('ul1');
var oLis = oUl.getElementsByTagName('li');

for (var i = 0; i < oLis.length; i++) {
    oLis[i].onmouseover = function () {
        this.style.backgroundColor = 'pink';
    };
    oLis[i].onmouseout = function () {
        this.style.backgroundColor = '';
    };
}

2.3、DOM深入:重绘与回流

  • 回流(也叫重排):当页面中的HTML结构发生改变(增加、删除元素、位置发生改变...),浏览器都需要重新计算一遍最新的DOM结构,重新对当前的页面进行渲染;

  • 重绘:某一个元素的部分样式发生改变了(背景颜色、字号等),浏览器只需要重新渲染当前页面即可;

2.4、动态创建节点再追加到页面的方式实现数据绑定

for (var i = 0; i < ary.length; i++) {
    var cur = ary[i];
    var oLi = document.createElement('li');
    oLi.innerHTML = '<span>' + i + '</span>' + cur.title + '---' + cur.desc;
    oUl.appendChild(oLi);
}

优势: 把需要动态绑定的内容一个个追加到页面中,对原来的元素没有任何的影响;

缺点:每当创建一个li,就添加到页面中,会引发一次DOM回流最后引发回流的次数过多,影响性能。

2.5、字符串拼接的方式实现数据绑定

var str = "";
for (var i = 0; i < ary.length; i++) {
    var cur = ary[i];
    str += '<li>';
    str += '<span>' + i + '</span>';
    str += cur.title;
    str += '</li>'
}
oUl.innerHTML += str;

首先循环需要绑定的数据,然后把需要动态绑定的标签以字符串的方式拼接到一起,拼接完成最后统一添加到页面中。

拼接完成的整体还是字符串,最后把字符串统一的添加到页面中,浏览器还需要把字符串渲染成最新的标签。

字符串拼接绑定数据,使以后工作中最常用的一种绑定数据的方式,因为所有模板引擎的数据绑定(jade、kTemplate....)以及所有的框架(angular.js、backbone.js....)的原理都是字符串拼接,所以说你可能没有自己直接使用字符串拼接,但是在使用引擎以及框架的过程已经用到了。

** 优势:** 事先把内容拼接好,最后统一添加到页面中,只引发一次回流;

缺点: 把新拼接的字符串添加到#ul1中,原有的三个li的鼠标滑过效果都消失了(原来标签绑定的事件都消失了)。

2.6、文档碎片的方式实现数据绑定

var frg = document.createDocumentFragment(); // 创建一个文档碎片,相当于临时创建了一个容器
for (var i = 0; i < ary.length; i++) {
    var cur = ary[i];
    var oLi = document.createElement('li');
    oLi.innerHTML = '<span>' + i + '</span>' + cur.title;
    frg.appendChild(oLi);
}
oUl.appendChild(frg);
frg = null;

这种方式解决了以上的问题,但是实际开发中用的很少。

三、表格排序

<ul id="ul1">
    <li>98</li>
    <li>99</li>
    <li>96</li>
    <li>95</li>
    <li>90</li>
</ul>
var utils = {
    listToArray:function (likeAry) {
        var ary = [];
        try {
            ary = Array.prototype.slice.call(likeAry);
        } catch (e) {
            for (var i = 0; i < likeAry.length; i++) {
                ary[ary.length] = likeAry[i];
            }
        }
        return ary;
    }
};
var oUl = document.getElementById('ul1');
var oLis = oUl.getElementsByTagName('li');

// 1、先把元素集合类数组转换为数组
var ary = utils.listToArray(oLis);

// 2、给数组进行排序:按照每一个li中的内容大小进行排序
ary.sort(function (a, b) {
    return parseFloat(a.innerHTML) - parseFloat(b.innerHTML);
});

// 3、按照ary中存储的最新顺序,依次的把对应的li添加到页面当中
var frg = document.createDocumentFragment();
for (var i = 0; i < ary.length; i++) {
    var obj = ary[i];
    frg.appendChild(obj);
}
oUl.appendChild(frg);
frg = null;

DOM映射机制:页面中的标签和JavaScript中获取到的元素对象或者元素集合是紧紧绑定在一起的,也中的HTML结构改变了,JS中不需要重新获取,集合里面的内容也会跟着自动改变。

var oUl = document.getElementById('ul1');
var oLis = oUl.getElementsByTagName('li');
console.log(oLis.length); // 5

var oLi = document.createElement('li');
oUl.appendChild(oLi);
console.log(oLis.length); // 6,没有重新的获取,但是oLis这个集合中的长度和内容会自动跟着发生改变

个人公众号(icemanFE):分享更多的前端技术和生活感悟

个人公众号.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,968评论 6 482
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,601评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 153,220评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,416评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,425评论 5 374
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,144评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,432评论 3 401
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,088评论 0 261
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,586评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,028评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,137评论 1 334
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,783评论 4 324
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,343评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,333评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,559评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,595评论 2 355
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,901评论 2 345

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,601评论 18 139
  • 第5章 引用类型(返回首页) 本章内容 使用对象 创建并操作数组 理解基本的JavaScript类型 使用基本类型...
    大学一百阅读 3,212评论 0 4
  • 自己的思维积累才是学霸养成的过程中最重要的步骤 就像通往财富自由之路中的注意力的重要程度一样,不同于时间、不同于金...
    晨听阅读 982评论 0 8
  • 问:请教一下老师,为什么都在说加息就意味着通胀要来了,这两者是怎样的逻辑关系.通胀要来,怎么黄金还一直跌.还有人说...
    ad5e912c526b阅读 453评论 0 0
  • 再相遇,你平淡如水,我亦云淡风轻。再交心,你轻描淡写,我亦含语不言。再牵手,你相安无事,我亦重头来过。
    叶初夏蔷薇阅读 180评论 0 0