1.数据类型
ECMAScript规范定义了7种数据类型
,分为基本类型和引用类型
基本类型:String,Boolean,Number,Undefined,Null, Symbol
引用类型:Object
基本类型:保存在栈内存中的简单数据段,按值访问,操作的是他们实际保存的值。
引用类型:保存在堆内存中的对象,意思是,变量中保存的实际上只是一个指针,这个指针指向内存堆中实际的值;按引用访问,当查询时,我们需要先从栈中读取内存地址,接着根据地址保存在堆内存中的值。
2. 判断JS数据类型的四种方法
(1)typeof
- typeof 操作符返回这个表达式的数据类型(全小写字母)
"string"—字符串
"boolean" —布尔值
"number"—数值
"undefined"—未定义或未初始化的变量
"symbol"— symbol值
"object"—对象或者null
"function"—函数
typeof " "; // string
typeof true; //boolean
typeof 1000; // number
typeof Symbol(); // symbol
typeof undefined; //undefined
typeof new Function(); // function
typeof null; //object
typeof [] ; //object
typeof new Date(); //object
总结一下规则
基本类型,除 null 以外,均可以返回正确的结果
引用类型,除 function 返回 function 类型以外,一律返回 object 类型
对于 null ,返回 object 类型
(2) instanceof
- instanceof 用来判断 A 是否为 B 的实例,表达式为:A instanceof B。特别注意的是:instanceof 检测的是原型,即 A的 proto 是否指向 B 的原型对象
{} instanceof Object;// true
[] instanceof Array; // true
[] instanceof Object; // true
function Person(){};
new Person() instanceof Person; //true
new Person instanceof Object;// true
instanceof 只能判断两个对象是否属于实例关系, 而不能判断一个对象实例具体属于哪种类型
,这也是我们很少使用instanceof 判断数据类型的原因。
instanceof 操作符的另外一个问题是:它假定只有一个全局执行环境。如果网页中包含多个框架,那实际上就存在两个以上不同的全局执行环境,从而存在两个以上不同版本的构造函数。如果你从一个框架向另一个框架传入一个数组,那么传入的数组与在第二个框架中原生创建的数组分别具有各自不同的构造函数。
(3) constructor
-
constructor 判断构造函数的方式
当一个函数 F被定义时,JS引擎会为F添加 prototype 原型,然后再在 prototype上添加一个 constructor 属性,并让其指向 F 的引用。
function F(){
}
var f=new F();
f.constructor ===F //true
//注意: constructor 在类继承时会出错
function A(){};
function B(){};
A.prototype = new B(); //A继承自B
var a = new A();
a.constructor === B // true;
a.constructor === A //false;
//其实我们可以修复constructor指向
//而instanceof方法不会出现该问题,对象直接继承和间接继承的都会报true:
a instanceof B //true;
a instanceof A //true;
细节问题:
null 和 undefined 是无效的对象,因此是不会有 constructor 存在的,这两种类型的数据需要通过其他方式来判断。
函数的 constructor 是不稳定的,这个主要体现在自定义对象上,当开发者重写 prototype 后,原有的 constructor 引用会丢失,constructor 会默认为 Object
(4)Object.prototype.toString.call()
- Obeject原型上的toString方法,该方法获取this对象上的[[Class]]属性,这个属性会返回该对象的类型,其格式为 [object Xxx] ,其中 Xxx 就是对象的类型
对于 Object 对象,直接调用 toString() 就能返回 [object Object] 。而对于其他对象,则需要通过 call / apply 来调用才能返回正确的类型信息。
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(new RegExp()) ; // [object RegExp]
Object.prototype.toString.call(new Error()) ; // [object Error]
Object.prototype.toString.call(document) ; // [object HTMLDocument]
Object.prototype.toString.call(window) ; //[object global] window 是全局对象 global 的引用
(5)使用特性判断
总结
通常我们只需要使用typeof + toString方法就可以识别全部数据类型