超越Jquery_01_isPlainObject分析与重构


Posted in Javascript onOctober 20, 2010

isPlainObject是Jquery1.4后提供的新方法,用于判断对象是否是纯粹的对象(通过 "{}" 或者 "new Object" 创建的)。

使用isPlainObject

首先我们来了解一下什么叫'纯粹的对象',简单的理解'纯粹的对象'指的就是由Object构造出来的对象。那哪些对象是由Object构造出来的呢。首当其充的肯定是由new Object()所构造出来的对象,注意:在Object后的括号里可没加任何东西。因为Object是所有'类'的根基,因此它有一些特殊的行为,如当调用new Object(3)的时候,会构造一个Number类型的对象。new Object('')会构造一个String类型的对象。然后以{}这种形式定义的对象也属于'纯粹的对象'。'{}'的实质就是new Object(),只是表现形形式不同。好,让我们来看一段代码:

var objStr = new Object(''); 
alert(objStr.constructor);//String 
alert(isPlainObject(objStr));//false 
var objNum = new Object(3); 
alert(objNum.constructor);//Number 
alert(isPlainObject(objNum));//false 
function Person(){} 
var person = new Person(); 
alert(isPlainObject(person));//false 
var obj01 = new Object(); 
obj01.name = '笨蛋的座右铭'; 
alert(isPlainObject(obj01));//true 
alert(isPlainObject({name:'笨蛋的座右铭'}));//true

isPlainObject源码分析
以下代码为Jquery中的isPlainObject的完整版本,注释已经很详尽了,我就不多说什么了。
var toString = Object.prototype.toString, 
hasOwnProperty = Object.prototype.hasOwnProperty; 
function isPlainObject( obj ) { 
// Must be an Object. 
// Because of IE, we also have to check the presence of the constructor property. 
//Make sure that DOM nodes and window objects don't pass through, as well 
//windows objects:toString.call(window):IE [object Object] FF [object Window] chrome [window global] safari [object DOMWindow] 
//DOM nodes:toString.call(#div01):IE [object Object] FF [object Window] chrome [object global] safari [object DOMWindow] 
//结论:obj.nodeType || obj.setInterval主要是针对于IE浏览器进行判断 
//注:history,location,navigator,screen的setInterval为undefined 
if ( !obj || toString.call(obj) !== "[object Object]" || obj.nodeType || obj.setInterval ) { 
return false; 
} 
// Not own constructor property must be Object 
// 除去自定义对象和内置对象的判断,如function Person(){} var p = new Person();String,Number 
if ( obj.constructor //有constructor属性 
&& !hasOwnProperty.call(obj, "constructor") //并且constructor这个属性必须是在原型链中进行定义的 
&& !hasOwnProperty.call(obj.constructor.prototype, "isPrototypeOf")//并且原型中有isPrototypeOf方法,一般只有Object的原型中才有这个方法 
) { 
return false; 
} 
// Own properties are enumerated firstly, so to speed up, 
// if last one is own, then all properties are own. 
//针对于复杂类结构,如有继承... 
/* 
//一个简单的测试 
function Animal(name){ 
} 
function Person(name,age){ 
Animal.call(this,name); 
this.age =age; 
} 
var p = new Person('jxl',20); 
for(key in p){ 
alert(hasOwnProperty.call( p, key ))//true , false 
} 
*/ 
var key; 
for ( key in obj ) {} 
return key === undefined || hasOwnProperty.call( obj, key ); 
}

提出问题
个人感觉这个实现比较复杂,而且有BUG。
简单的BUG,history,location,navigator,screen可以顺序通过 isPlainObject的检测返回true.
来看一个我的解决方案(修改BUG,简化):
function isPlainObject(obj){ 
if(obj&&Object.prototype.toString.call(obj)==="[object Object]"&&obj.constructor===Object &&!hasOwnProperty.call(obj, "constructor")){ 
var key; 
for ( key in obj ) {} 
return key === undefined || hasOwnProperty.call( obj, key ); 
} 
return false; 
}

还有BUG,而且是一个无解的BUG:
function m(){}; 
m.prototype.constructor=Object; //必杀 
obj=new m; 
alert(isPlainObject(obj)); //true

再来一个同理的:
function m(){}; 
m.prototype = {}; 
obj=new m; 
alert(isPlainObject(obj)); //true

这个答案是无解的!

解答无解
本以为这个问题很好解决,结果深入后,发现这是一个无解的问题。原因如下:

function Person(){}; 
Person.prototype.constructor=Object; 
var person=new Person;

让我们来看一下person现在的状态:
超越Jquery_01_isPlainObject分析与重构
person和其构造函数Person唯一的联系就是其prototype链中的constructor属性。而在我们判断是否为'纯粹的对象'主要是依据对象实例的constructor进行的。如果我们将其指向Object,正如图中看到的那样,那么person和Person在代码上就没有关系了。也正是因为这一点,让类型的判断出现了问题。
Javascript 相关文章推荐
javascript获取URL参数与参数值的示例代码
Dec 20 Javascript
jQuery异步验证用户名是否存在示例代码
May 21 Javascript
jQuery中extend函数详解
Feb 13 Javascript
AngularJs 弹出模态框(model)
Apr 07 Javascript
JS实现的简单轮播图运动效果示例
Dec 22 Javascript
react-native-fs实现文件下载、文本存储的示例代码
Sep 22 Javascript
bootstrapTable+ajax加载数据 refresh更新数据
Aug 31 Javascript
JS实现点击拉拽轮播图pc端移动端适配
Sep 05 Javascript
jQuery each和js forEach用法比较
Feb 27 jQuery
React中获取数据的3种方法及优缺点
Feb 18 Javascript
vue如何实现关闭对话框后刷新列表
Apr 08 Vue.js
vue项目配置sass及引入外部scss文件
Apr 14 Vue.js
理解Javascript_15_作用域分配与变量访问规则,再送个闭包
Oct 20 #Javascript
理解Javascript_14_函数形式参数与arguments
Oct 20 #Javascript
理解Javascript_13_执行模型详解
Oct 20 #Javascript
用jquery与css打造个性化的单选框和复选框
Oct 20 #Javascript
Jquery插件之多图片异步上传
Oct 20 #Javascript
jquery判断checkbox(复选框)是否被选中的代码
Oct 20 #Javascript
jQuery下扩展插件和拓展函数的写法(匿名函数使用的典型例子)
Oct 20 #Javascript
You might like
PHP加速 eAccelerator配置和使用指南
2009/06/05 PHP
php中关于codeigniter的xmlrpc的类在进行数据交换时的类型问题
2011/07/03 PHP
php获取远程图片的两种 CURL方式和sockets方式获取远程图片
2011/11/07 PHP
laravel 4安装及入门图文教程
2014/10/29 PHP
PHP跨平台获取服务器IP地址自定义函数分享
2014/12/29 PHP
PHP使用NuSOAP调用Web服务的方法
2015/07/18 PHP
PHP多维数组转一维数组的简单实现方法
2015/12/23 PHP
PHP中字符串长度的截取用法示例
2017/01/12 PHP
js 单引号 传递方法
2009/06/22 Javascript
Array.prototype.slice.apply的使用方法
2010/03/17 Javascript
Jquery公告滚动+AJAX后台得到数据
2011/04/14 Javascript
jQuery实现判断滚动条到底部
2015/06/23 Javascript
分享jQuery封装好的一些常用操作
2016/07/28 Javascript
JS实现的表格行上下移动操作示例
2016/08/03 Javascript
深入理解AngularJS中的ng-bind-html指令和$sce服务
2016/09/08 Javascript
nodejs实现邮件发送服务实例分享
2017/03/29 NodeJs
Vue实现动态显示textarea剩余字数
2017/05/22 Javascript
webpack2.0配置postcss-loader的方法
2017/08/17 Javascript
深入理解es6块级作用域的使用
2019/03/28 Javascript
jQuery 筛选器简单操作示例
2019/10/02 jQuery
javascript实现前端成语点击验证
2020/06/24 Javascript
微信小程序实现上传照片代码实例解析
2020/08/04 Javascript
[01:38]女王驾到——至宝魔廷新尊技能&特效展示
2020/06/16 DOTA
简单谈谈Python中的反转字符串问题
2016/10/24 Python
浅谈Python2.6和Python3.0中八进制数字表示的区别
2017/04/28 Python
浅谈python内置变量-reversed(seq)
2017/06/21 Python
使用50行Python代码从零开始实现一个AI平衡小游戏
2018/11/21 Python
详解重置Django migration的常见方式
2019/02/15 Python
Python使用requests xpath 并开启多线程爬取西刺代理ip实例
2020/03/06 Python
python 实现有道翻译功能
2021/02/26 Python
一站式跨境收款解决方案:Payoneer(派安盈)
2018/09/06 全球购物
Abbacino官网:包、钱包和女士配饰
2019/04/15 全球购物
Liu Jo西班牙官网:意大利服装品牌
2019/09/11 全球购物
美国新娘礼品店:The Paisley Box
2020/09/08 全球购物
家长对孩子评语
2014/01/30 职场文书
玩手机检讨书1000字
2014/10/20 职场文书