转载来源 https://www.cnblogs.com/echolun/p/10287616.html

一、前言

如何判断一个对象或一个值是否是一个数组,在面试或工作中我们常常会遇到这个问题,既然出现频率高,想着还是做个整理,那么本文主要基于几种判断方式,以及方式判断的原理,是否存在问题展开讨论。

二、判断对象是否是数组的几种方式

1.通过instanceof判断

instanceof运算符用于检验构造函数的prototype属性是否出现在对象的原型链中的任何位置,返回一个布尔值。
let a = [];
a instanceof Array; //true
let b = {};
b instanceof Array; //false

在上方代码中,instanceof运算符检测Array.prototype属性是否存在于变量a的原型链上,显然a是一个数组,拥有Array.prototype属性,所以为true。

存在问题:

需要注意的是,prototype属性是可以修改的,所以并不是最初判断为true就一定永远为真。

其次,当我们的脚本拥有多个全局环境,例如html中拥有多个iframe对象,instanceof的验证结果可能不会符合预期,例如:

//为body创建并添加一个iframe对象
var iframe = document.createElement('iframe');
document.body.appendChild(iframe);
//取得iframe对象的构造数组方法
xArray = window.frames[0].Array;
//通过构造函数获取一个实例
var arr = new xArray(1,2,3);
arr instanceof Array;//false

导致这种问题是因为iframe会产生新的全局环境,它也会拥有自己的Array.prototype属性,让不同环境下的属性相同很明显是不安全的做法,所以Array.prototype !== window.frames[0].Array.prototype,想要arr instanceof Array为true,你得保证arr是由原始Array构造函数创建时才可行。

2.通过constructor判断

我们知道,实例的构造函数属性constructor指向构造函数,那么通过constructor属性也可以判断是否为一个数组。
let a = [1,3,4];
a.constructor === Array;//true

同样,这种判断也会存在多个全局环境的问题,导致的问题与instanceof相同。

//为body创建并添加一个iframe标签
var iframe = document.createElement('iframe');
document.body.appendChild(iframe);
//取得iframe对象的构造数组方法
xArray = window.frames[window.frames.length-1].Array;
//通过构造函数获取一个实例
var arr = new xArray(1,2,3);
arr.constructor === Array;//false

3.通过Object.prototype.toString.call()判断

Object.prototype.toString().call()可以获取到对象的不同类型,例如

let a = [1,2,3]
Object.prototype.toString.call(a) === '[object Array]';//true

它强大的地方在于不仅仅可以检验是否为数组,比如是否是一个函数,是否是数字等等

//检验是否是函数
let a = function () {};
Object.prototype.toString.call(a) === '[object Function]';//true
//检验是否是数字
let b = 1;
Object.prototype.toString.call(a) === '[object Number]';//true

甚至对于多全局环境时, Object.prototype.toString().call()也能符合预期处理判断。

//为body创建并添加一个iframe标签
var iframe = document.createElement('iframe');
document.body.appendChild(iframe);
//取得iframe对象的构造数组方法
xArray = window.frames[window.frames.length-1].Array;
//通过构造函数获取一个实例
var arr = new xArray(1,2,3);
console.log(Object.prototype.toString.call(arr) === '[object Array]');//true

4.通过Array.isArray()判断

Array.isArray() 用于确定传递的值是否是一个数组,返回一个布尔值。

let a = [1,2,3]
Array.isArray(a);//true

简单好用,而且对于多全局环境,Array.isArray() 同样能准确判断,但有个问题,Array.isArray() 是在ES5中提出,也就是说在ES5之前可能会存在不支持此方法的情况。怎么解决呢?

三、判断数组方法的最终推荐

 当然还是用Array.isArray(),从ES5新增isArray()方法正是为了提供一个稳定可用的数组判断方法,不可能专门为此提出的好东西不用,而对于ES5之前不支持此方法的问题,我们其实可以做好兼容进行自行封装,像这样:
if (!Array.isArray) {
Array.isArray = function(arg) {
return Object.prototype.toString.call(arg) === '[object Array]';
};
}

那么对于数组判断的几种方式也说完了,合理的推荐也给出了,有什么问题或者错误的地方欢迎大家支持

参考资料:

Determining with absolute accuracy whether or not a JavaScript object is an array

最新文章

  1. iOS开发之功能模块--根据需求开发横向的子弹盒View
  2. (转)PhoneGap开发环境搭建
  3. 关于Model层中Datetime Datetime? 默认值的问题
  4. 【zoj3254】Secret Code
  5. MyBatis自动获取主键,MyBatis使用Oracle返回主键,Oracle获取主键
  6. vsftpd的主配置文件详解
  7. js中原生对象、内置对象和宿主对象(转)
  8. C# 6 与 .NET Core 1.0 高级编程 - 40 ASP.NET Core(下)
  9. 从数组中每次取一个不同的数组成员 getRandomItem(arr)
  10. MySQL之事务的四大特性
  11. Windows脚本相关
  12. Linux系统把/home重新挂载到其他硬盘或分区
  13. Jmeter-----【mac电脑】配置web浏览器的代理抓取请求
  14. sparkStreaming消费kafka-1.0.1方式:direct方式(存储offset到zookeeper)-- 2
  15. Python中识别DataFrame中的nan
  16. Netty---入门程序,搭建Websocket 服务器
  17. 34-BigInteger详解
  18. javascript运算符instanceof
  19. Java随机获取32位密码且必须包含大小写字母、数字和特殊字符,四种的任意三种
  20. HDU 6148 Valley Numer (数位DP)题解

热门文章

  1. emacs第二天
  2. java 多态调用
  3. classmethode,staticmethode、反射
  4. redhat quay 安装试用
  5. iptables 自定义链
  6. mysql(六)数据库连接池
  7. 【医学】三分钟看懂乳腺BI-RADS分级
  8. rpm续
  9. 基于ELK 7.50搭建elastalert 监控报警和权限控制
  10. json字符串手动拼接