超全Javascript数据类型检测方式超全Javascript数据类型检测 JavaScript中为了准确判断一个变量
超全Javascript数据类型检测
JavaScript中为了准确判断一个变量的类型,可以有多种方法。下面是一些JavaScript数据类型检测方法:
一、 typeof
-
检测基本数据类型: 除了null以外都能返回正常的值
-
检测引用类型: 除function外都返回'object'
// 基本类型除了null typeof 123 // 'number' typeof 'abc' // 'string' typeof true // 'boolean' typeof undefined // 'undefined' typeof NaN // 'number' 注意! typeof null // 'object' 注意!!!
// 引用类型 typeof {} // 'object' typeof [] // 'object' typeof function(){} // 'function' typeof RegExp // 'function' 注意正则返回function
注意:一般用于基本类型检测(除null)
二、 instanceof: 是用来判断 A 是否为 B 的实例(检测引用类型)
instanceof
是基于原型链来工作的,它不仅仅检查对象
是否直接由某个构造函数创建,还会检查对象是否继承自该构造函数的原型。
[] instanceof Array; //true
[] instanceof Object; //true 注意注意
{} instanceof Object; //true
new Date() instanceof Date; //true
"123" instanceof String; //false
new String('123') instanceof String; // true
- null 和 undefined:
null
和undefined
是原始数据类型,它们没有原型链,因此null instanceof Anything
和undefined instanceof Anything
都会返回false
。 - 基本数据类型:对于字符串、数字、布尔值等基本数据类型,它们不是对象,因此没有原型链,所以
instanceof
对这些类型无效,总是返回false
。 - 修改
prototype
和constructor
:由于instanceof
依赖于prototype
和constructor
属性,因此如果这些属性被修改/重写,可能会影响instanceof
的结果。
三、 constructor: 本意是获取构造函数/获取实例所属的类
每个对象都有一个
constructor
属性,该属性指向创建该对象的构造函数的引用。所以我们可以使用constructor
来间接判断当前对象的类型。
[].constructor === Array; //true
[].constructor === Object; // false
{}.constructor === Object; //true
RegExp.constructor === Function; // true
"123".constructor === String; //true 注意!!不推荐
true.constructor === Boolean; //true 注意!! 不推荐
1.constructor === Number // 报错 注意!! 不推荐
(1).constructor === Number // true 不推荐
注意:
- null和undefined 没有constructor,不能用其判断;
- constructor不稳定,,因为开发者开可能会把类的原型进行重写, 这样检测出来的结果就是不准确的。
- 基本数据类型一般不使用constructor进行检测,而是使用typeof。
"123".constructor === String; // true 虽然浏览器控制台会返回true, 但不代表所有js环境下运行会正常,其他环境可能抛出错误。浏览器的控制台中没有报错,可能是因为该环境在后台对原始值进行了自动包装的结果。
四、Object.prototype.toString.call(value)
用来检测数据类型比较准确。
Object.prototype.toString()
方法返回的是一个表示该对象的内部 [[Class]] 属性的字符串,这个字符串对于不同的数据类型是唯一的,即使对象的自定义toString()
方法被重写,也不会影响到Object.prototype.toString
的行为。
Object.prototype.toString.call('12'); // [object String]
Object.prototype.toString.call(112) ; // [object Number]
Object.prototype.toString.call(false); // [object Boolean]
Object.prototype.toString.call(new Function()); // [object Function]
Object.prototype.toString.call(new Date()); // [object Date]
Object.prototype.toString.call([]); // [object Array]
Object.prototype.toString.call(document)); // [object HTMLDocument]
Object.prototype.toString.call(window); //[object global] window 是全局对象 global 的引用
Object.prototype.toString.call(undefined); // [object Undefined] 注意!!
Object.prototype.toString.call(null); // [object Null] 注意!!
五、其他-实用场景
1. 数组检测:
- [] instanceof Array; //true (若被改写,可能会不准确)
- [].constructor == Array; //true (若被改写,可能会不准确)
- Object.prototype.toString.call([]); // [object Array]
- Array.isArray([]) // true
2. 空数组检测:
- Array.isArray(arr) && arr.length === 0
- JOSN.stingify(arr) === '[]'
3. 对象检测:
- {}.constructor == Object; //true (若被改写,可能会不准确)
- Object.prototype.toString.call({}); // [object Object]
4. 空对象检测:
-
Object.prototype.toString.call(obj) === " [object Object ]" && Object.keys(obj).length === 0
-
JOSN.stingify(obj) === '{}'
注意,JOSN.stingify(arr) === '[]'、JOSN.stingify(obj) === '{}' 适合体量小,数组或对象中不存在循环引用的情况。
5. NaN检测
isNaN()
方法:
- ES5方法,用于检查给定的值是否为NaN。
- 会尝试将参数转换为数字。如果参数不能被转换为数字,或者转换后的结果为NaN,则
isNaN()
会返回true
。
Number.isNaN()
方法:
- ES6引入的方法,用于判断一个值是否严格等于NaN。
- 不会尝试将传入的参数转换为数字。如果参数的值和类型都是NaN,则返回
true
;否则返回false
。
-
例子:
console.log(isNaN(NaN)); // true console.log(isNaN(0 / 0)); // true console.log(isNaN("NaN")); // true,注意这里!字符串"NaN"被转换成了数字NaN console.log(isNaN("123")); // false,因为"123"被转换成了数字123 console.log(isNaN("Hello")); // true,因为"Hello"不能被转换为有效的数字 console.log(isNaN(true)); // false,因为true被转换成了数字1 console.log(isNaN(undefined)); // true,因为undefined被转换成了NaN console.log(isNaN({})); // true,因为对象被转换成了NaN(实际上是先调用对象的valueOf()或toString(),如果返回的是非原始值则继续尝试,直到得到原始值为止,但对象的情况比较复杂,通常会导致转换为NaN) console.log(Number.isNaN(NaN)); // true console.log(Number.isNaN(0 / 0)); // true console.log(Number.isNaN("NaN")); // false 不进行类型转换 console.log(Number.isNaN("123")); // false 不进行类型转换 console.log(Number.isNaN("Hello")); // false 不进行类型转换 console.log(Number.isNaN(true)); // false 不进行类型转换 console.log(Number.isNaN(undefined)); // false 不进行类型转换 console.log(Number.isNaN({})); // false 不进行类型转换
转载自:https://juejin.cn/post/7418109561108381706