整理JS如何判断数据类型
5views
总体可以分为这几类:
typeof
instanceof
Object.prototype.toString()
constructor
// 初始化一些变量
var num = 123;
var str = 'abcdef';
var bool = true;
var arr = [1, 2, 3, 4];
var obj = {name:'wenzi', age:25};
var func = function(){ console.log('this is function'); }
var und = undefined;
var nul = null;
var date = new Date();
var reg = /^[a-zA-Z]{5,20}$/;
var error= new Error();
typeof num, // number
typeof str, // string
typeof bool, // boolean
typeof arr, // object
typeof obj, // object
typeof func, // function
typeof und, // undefined
typeof nul, // object
typeof date, // object
typeof reg, // object
typeof error // object
可以看出 arr
、json
、null
、date
、reg
、error
全部被检测为object
。因此其只能适用于基本数据类型。
num instanceof Number, // false
str instanceof String, // false
bool instanceof Boolean, // false
arr instanceof Array, // true--注意
arr instanceof Object, // true--注意
obj instanceof Object, // true--注意
func instanceof Function, // true
und instanceof Object, // false
nul instanceof Object, // false
date instanceof Date, // true
reg instanceof RegExp, // true
error instanceof Error // true
虽然instanceof
可以判断引用类型,但是注意arr
和obj
在instanceof Object
都是true
,会导致判断是对象时不准确。
Object.prototype.toString.call(num), // '[object Number]'
Object.prototype.toString.call(str), // '[object String]'
Object.prototype.toString.call(bool), // '[object Boolean]'
Object.prototype.toString.call(arr), // '[object Array]'
Object.prototype.toString.call(obj), // '[object Object]'
Object.prototype.toString.call(func), // '[object Function]'
Object.prototype.toString.call(und), // '[object Undefined]'
Object.prototype.toString.call(nul), // '[object Null]'
Object.prototype.toString.call(date), // '[object Date]'
Object.prototype.toString.call(reg), // '[object RegExp]'
Object.prototype.toString.call(error) // '[object Error]'
function Person(){}
var Tom = new Person();
// undefined和null没有constructor属性
console.log(
Tom.constructor==Person,
num.constructor==Number,
str.constructor==String,
obj.constructor==Boolean,
arr.constructor==Array,
json.constructor==Object,
func.constructor==Function,
date.constructor==Date,
reg.constructor==RegExp,
error.constructor==Error
);
// 所有结果均为true
constructor本来是原型对象上的属性,指向构造函数。但是根据实例对象寻找属性的顺序,若实例对象上没有实例属性或方法时,就去原型链上寻找,因此,实例对象也是能使用constructor属性的。
但仍有两个问题:
- undefined和null没有constructor属性,所以判断时代码可能会报错。
- 由于constructor属性是可以变更的,也会导致检测出的结果不正确
但是以上方法都不严谨,可能遇到如下问题:
const obj = { [Symbol("a")]: 1 }
console.log(Object.keys(obj))
console.log(JSON.stringify(obj))
console.log(Reflect.ownKeys(obj))
运行打印下:
[]
{}
[ Symbol(a) ]
因此更严谨的方法是:Reflect.ownKeys(obj).length === 0
最常用,可以在不同的窗口(window)和框架(iframe)中正常工作,不受上下文环境的限制。
简单易懂,性能较好,适用于所有标准的 JavaScript 环境,具有很好的兼容性。
会存在误判:
function MyArray() {// 自定义的类似数组的对象this.length = 0;}
MyArray.prototype = Array.prototype;const myArr = new MyArray();console.log(myArr instanceof Array); // 输出: true
会存在误判:
function isArray(obj) {return obj.constructor === Array;}
// 示例用法
let obj = { name: "John", age: 25 };console.log(isArray(obj)); // 输出:false
obj.constructor = Array;console.log(isArray(obj)); // 输出:true