《献给你,我深爱的ECMAScript》之Object篇
本文主要想说说ECMAScript中的Object相关的东西,主要内容会定位在5和6。
题外话:
其实我自己博客一直维护着一个分类叫《js-object》,里面收录了一些框架关于Object扩展的业务api和ECMAScript已有以及新增相关. 本文中部分内容也来自这个分类,有兴趣的可以看看。
我们先看一段代码:
/* prototype 1.6.1 * 这个版本很简单: * 没有hasOwnProperty的过滤,采用push * 但是prototype的代码命名很正统,语义化意识明显 */ function keys(object){ var results = []; for(var property in object){ results.push(property); } return results; } /* prototype 1.7 * 辅助函数 -- 其实如果自己抽离的话,就不需要这样设计 * 对参数进行了类型验证 * 加入了hasOwnProperty过滤 */ function Type(o){ switch(o){ case null : return NULL_TYPE; case (void 0) : return UNDEFINED_TYPE; } var type = typeof o; switch(type){ case 'boolean' : return BOOLEAN_TYPE; case 'number' : return NUMBER_TYPE; case 'string' : return STRING_TYPE; } return OBJECT_TYPE; } function keys(object){ //加了一层类型判定 if(Type(object) !== OBJECT_TYPE){ throw new TypeError(); } var results = []; for(var property in object){ //改进了hasOwnProperty if(object.hasOwnProperty(property)){ results.push(property); } } return results; } /* underscore 1.5,1 * 加入了ES5的兼容分支,同时也有参数类型验证 * 也辅助函数_.has的过滤 */ _.has = function(obj,key){ Object.prototype.hasOwnProperty.call(obj,key); }; var nativeKeys = Object.keys; _.keys = nativeKeys || function(obj){ if(obj !== Object(obj)){ throw new TypeError('Invalid object'); } var keys = []; for(var key in obj){ if(_.has(obj,key)){ keys.push(key); } } return keys; }; /* kissy 1.3 * 这个和MDN上差不多,加了一层hasEnumBug的判定 */ var hasEnumBug = !({toString:1}.propertyIsEnumerable('toString')), enumProperties = [ "constructor", "hasOwnProperty", "isPrototypeof", "propertyIsEnumerable", "toString", "toLocaleString", "valueOf" ]; keys:function(o){ //一次定义变量 var result = [], p, i; for(p in o){ result.push(p); } if(hasEnumBug){ for(i = enumProperties.length-1;i>=0;i--){ p = enumProperties[i]; if(o.hasOwnProperty(p)){ result.push(p); } } } return result; }; /* tangram 2.0.2.4 * 没有采用push,而是用length++的方式装载 */ baidu.object.keys = function(source){ //参数一次定义 var result = [], resultLen = 0, k; for(k in source){ if(source.hasOwnProperty(k)){ //没有使用push result[resultLen++] = k; } } }; /* qwrap 1.1.5 * 比较标准,过滤了hasOwnProperty */ keys:function(obj){ var ret = []; for(var key in obj){ if(obj.hasOwnProperty(key)){ ret.push(key); } } return ret; };
代码有点多,但是我相信很多同学应该都能从里面看出点东西~
1. 首先解释一下这个方法是干嘛的?
就是从一个Object类型的变量里面取它所有的keys,然后返回一个包含所有keys的数组。
2. 关于各大框架的罗列?
这个是很有心思的,大致是我个人研习的框架的顺序,并不是一般人认为的”远离国产,首选外国货“。
在这里小春真心感谢所有以上源码的作者们~
3. 为什么还标注版本?
我承认我是一个标准的源码版本控,当然这个在我研习各大框架源码实习细节上有很大的帮助,让我看到了作者对一些api的优化过程,从而自我反思一些东西。
这点如果看过我博客的里面介绍《jquery数据存储》的同学,应该有一定的体会。
好吧,入正题:
其实想从现有的一些框架源码设计的细节角度,去看看对ECMAScript Object的支持情况。
每次说到js,都会被问道一个很基本的问题,这个针对低版本浏览器不?
关注兼容性的同学可以猛戳: 这里
我们可以看到,其实大致: IE 9+ FF 4+ Chrome 5+ 是支持的,IE6-8目前是不支持的,但是从上面的源码展示里面我们看到underscore是双分支兼容的。
我个人是非常坚持在类库设计的时候来更新这些原生方法的,原因很简单: 浏览器本身支持的方法“基本上”是最快的,当然以前的数组操作还是有一些差异的,不过高级浏览器也优化了部分。
其他一些细节简单地回顾一下:
如何判定参数是Object类型:
_.isObject = function(obj){ return obj === Object(obj); }; //tangram 2.0.0.0 //与原版本参数命名不一样,个人不推荐unknow baidu.isObject = function(source){ return typeof source === "function" || (typeof source === "object" && source != null) }; //qwrap 1.1.5 isObject:function(obj){ return obj !== null && typeof obj == 'object' };
很多关注ECMAScript的同学应该和我一样,会有一个问题,既然有keys了,为啥不直接再来一个values呢?
其实它出现过,不是在ES5里面,而是在Ees3.1的一个官方wiki讨论里面
当然也有人在bugzilla讨论Object.keys的时候,给了一个 方案
//有的外国人就是不喜欢{}这东西 Object.values = function (O) Object.keys(O).map(function(k) O[k]);
那我们如何应该是values呢?
/* underscore1.4.1 * https://github.com/jashkenas/underscore/blob/1.4.1/underscore.js * 依赖_.has的过滤,还是push的方式 */ _.values = function(obj){ var values = []; for(var key in obj){ if(_.has(obj,key)){ //只是存的东西变成了obj[key] value.push(obj[key]); } } return values; }; /* underscore 1.5.1第一个版本 * 依赖_.keys,先取出keys,然后按照keys的length进行for-in */ _.values = function(obj){ //一次定义变量 var keys = _.keys(obj), values = [], i = 0, length = keys.length; for(;i < length;i++){ values[i] = obj[key[i]]; } return values; }; //2013-7-24 braddunbar完善了一把 _.values = function(obj){ var keys = _.keys(obj), length = keys.length, //直接定义一个有长度的Array,而不是[] values = new Array(length), i = 0; for(;i < length;i++){ values[i] = obj[keys[i]]; } return values; }; /* tangram 2.0.0.0 * 而是不采用push方式,length++装载值 */ baidu.object.values = function(source){ //一次定义变量 var result = [], resultlen = 0, k; for(k in source){ if(source.hasOwnProperty(k)){ result[resultLen++] = source[k]; } } }; /* qwrap 1.1.5 * 一如既往的朴实风格,和keys基本代码一致,就是装载的东西变了一下下 */ values: function(obj){ var ret = []; for(var key in obj){ if(obj.hasOwnProperty(key)){ //和keys实现基本一样,只是push的对象不一样 ret.push(obj[key]); } } return ret; };
简单提提kissy:
关于kissy,我个人是一个长期源码观察者,除了代码设计层面给我的一些启发外, 还有一些小细节:
比如关于某一个方法在某些操作系统下的浏览器支不支持的注释, 让我比较佩服,
毕竟测试覆盖到如此之细~ 比如:kissy 1.3 的3474行:ie 8.0.7600.16315@win7 json bug!
所以针对一些源码source版本很干净的,注释很少的,我基本会“适度”研习而之丢弃。
其实对于Object,我们以前在频繁操作它的时候有很多痛: 比如:看看它的长度,也得遍历一遍,取出keys,看keys数组的长度来知道这个Object的长度。
下面是underscore 1.5.1的代码:
/* underscore 1.5.1 * return the number of values in the list * 参数过滤,支持Array和Object */ _.size = function(obj){ //参数过滤,如果null或者undefined,返回0 if(obj == null){ return 0; } //支持Array类型,方式还是比较独特的 //如果是Array,直接length //如果是Object,取keys,这个方法上面已经给出 return (obj.length === +obj.length) ? obj.length : _.keys(obj).length; } //示例: _.size({name:"zhangyaochun",job:"fe",for:"w3cplus"}); //3 _.size([1,2,3]); //3
这个方法,返回一个Object或者Array的长度。
再看看ECMAScript6 这次给出新的方法~
Object.is
用来判定两个值是否相等
语法:
var isSame = Object.is(value1,value2);
我们先来看看代码示例:
Object.is("1",1); //false Object.is("name","name"); //true Object.is(0,-0); //false Object.is(NaN,NaN); //true Object.is(NaN,0/0); //true
基本上,功能和===差不多,但是有特例:0 和 -0
用心一点的同学是不是发现:这个方法也可以用来判定是否是NaN嘛
下面是提供的一种兼容方式:
if(!Object.is){ Object.is = function(v1,v2){ if(v1 === 0 && v2 === 0){ return 1 / v1 === 1 / v2; } if(v1 !== v1){ return v2 !== v2; } return v1 === v2; }; }
小知识点:
braddunbar 这个人有人认识不?
他自己的博客描述:
Hey, I'm Brad,a husband,dad,javascripter and enjoyer of peanut butter sandwiches (preferably with honey and raisins).
其实他和undersorce、backbone、jquery、jquery ui都有关系。
在我眼中,他虽然只是一个contributor,但他是一个值得尊敬的人~
扩展阅读:
如需转载,烦请注明出处:http://www.w3cplus.com/js/ecmascript-lesson-2.html