原生JS篇三、数据类型及类型检测

本篇文章内容:

  • ECMAScript数据类型有哪些
  • 基本数据类型(也叫简单数据类型)中的undefined和null的区别与联系
  • typeOf()函数判断基本数据类型
  • instanceof用于判断引用数据类型
  • Object.prototype.toString.call()检测数据类型的通用方法
  • 扩展constructor属性检测数据类型方式以及数组进行类型检测时需注意的事项

ECMAScript有几种数据类型?

ECMAScript中有5种简单数据类型(也称为基本数据类型):undefined、Null、Boolean、Number、String;
有一种复杂数据类型:Object;

Undefined

Undefined类型只有一个值,即undefined.
使用var声明变量但是未进行初始化时,这个变量的值就是undefined.

Null

Null类型同样只有一个值,就是null.
null表示一个空对象指针,当定义的一个变量准备将来保存对象,那么最好将该变量初始化为null。

面试时也有遇到过undefined和null的区别或者说联系的问题,对于这个问题就可以这样回答:
实际上undefined值是派生自null值的,因此ECMA-262规定它们的相等性测试要返回true,即null == undefined //true。这就是null和undefined这两个值之间的联系,但是两者也有明显的区别。undefined的作用是当声明了一个变量但还尚未定义的时候,这个变量就默认取undefined值,但无论什么情况下都没必要将一个变量显示的设置为undefined,null则不同,只要当用于保存对象的变量还未保存对象时,都应该将其初始化为null.
*需要注意的是,虽然null == undefined的结果为true,因为两者是类似的值,但是null === undefined 的结果为false,因为两者类型不同。

Boolean

该类型有两个字面值:true和false.

Number

区别于其他语言,ECMAScript并没有为整数和浮点数分别定义数据类型,Number类型用于表示所有数值。

String

即字符串。

Object

该类型是这门语言中所有对象的基础类型。
javascript中的引用类型有:Object、Function、Date、Array、RegExp、Boolean、Number、String和自定义类等,但是所有这些引用类型都是对象,即Object。
另外在javascript高级程序设计中有这样一句话:引用类型的值(对象)是引用类型的一个实例。在ECMAScript中,引用类型是一种数据结构,用于将数据和功能组织在一起。

认识了javascript的数据类型之后,我们来看它们的类型检测问题,如何判断一个数据的类型呢?

首先我们来细数一下javascript中的数据类型,我们将它们分为两类:
基本数据类型:String,Number,Boolean,Undefined,Null;
引用类型:Function、Array、Object、Date、RegExp和自定义对象等。

下面分别介绍三种方式来判断数据类型:

typeOf()函数判断基本数据类型

对于基本数据类型,我们可以用typeOf()函数来判断类型,但有一个例外就是null,它是不能用typeOf()来判断的。

typeof(1); // number
typeof('1'); // string
typeof(true); // boolean
typeof(undefined); // boolean
typeof(null); // object
typeof(console.log); // function  

可以看到执行typeof()函数可以得到六种结果,分别是number、string、boolean、oject、function,其中基本类型数据除了null都能通过typeof()得到验证,所以检测出null以外的基本函数都可以用typeof来进行类型检测。

另:为什么typeof(null)值为object?
实际上是javascript最初实现中的一个错误,然后被EAMAScript沿用了,现在null被认为是对象的占位符,从而解释了这一矛盾。

instanceof用于判断引用数据类型

typeof()函数对于基本数据类型的判断还差强人意,但是它是完全没办法用于判断引用数据类型的,因为所有引用数据类型都会返回Object。于是javascript引入了instanceof,用来判断一个变量是某个对象的实例。

var obj = {}
console.log(obj instanceof Object); // true
var arr = []
console.log(arr instanceof Array); // true
var date = new Date()
console.log(date instanceof Date); // true
var fn = function() {}
console.log(fn instanceof Function); // true
var str = '123'
console.log(str instanceof String); // false

通过以上的代码可见,instanceof对于引用类型的类型检测支持很好,但是无法对基本类型数据进行类型检测。

Object.prototype.toString.call()

在javascript高级程序设计中提供了一种可以通用的来判断基本数据类型和引用数据类型的方法。

//1、对基本数据类型num1和对基本包装类型(特殊的引用类型)num2的类型检测            
var num1 = 1;
var num2 = new Number(1);
console.log(Object.prototype.toString.call(num1) == '[Object Number]'); // true
console.log(Object.prototype.toString.call(num2) == '[Object Number]'); // true
//2、对数组的类型检测
var arr = [];
console.log(Object.prototype.toString.call(arr) == '[Object Array]') // true
//3、对函数的类型检测
var fn = function() {}
console.log(Object.prototype.toString.call(fn) == '[Object Function]') // true
//4、对自定义对象的类型检测
function A() {}
var a = new A();
console.log(Object.prototype.toString.call(a) == '[Object Object]') // true

由此可见,Object.prototype.toString.call()可以用来作为自定义类以外的数据类型检测的通用方法,而对于自定义类我们可以转而使用instanceof来进行判断。

Object.prototype.toString.call()做类型判断的结果情况如下:

  1. 字符串:'[Object String]'
  2. 数字:'[Object Number]'
  3. 布尔值:'[Object Boolean]'
  4. undefined:'[Object Undefined]'
  5. null:'[Object Null]'
  6. 数组:'[Object Array]'
  7. 正则表达式:'{Object RegExp}'
  8. 日期对象:'[Object Date]'

constructor属性在类型检测中的应用以及数组类型检测注意事项

constructor是Object的每个实例都存在的一个属性(再强调一遍,在ECMAScript中,Object类型是所有其他实例的基础,换句话说Object类型所具有的属性和方法也存在于更具体的对象中),该属性(constructor)保存着用于创建当前对象的函数,即该属性会返回对创建对象的构造函数的引用。

[].constructor == Array // true
{}.constructor == Object // true

需要注意的是,当用instanceof和constructor对数组进行类型检测时,被判断的Array必须是当前页面声明的:
比如一个页面(父页面)有一个框架(iframe),框架中引用了一个页面(子页面),在子页面中声明了一个Array,并将其赋值给父页面的一个变量,Array == Object.constructor;会返回false.
原因是:
1、Array属于引用类型数据,在传递的时候仅仅是地址的传递。
2、每个页面的Array原生对象所引用的地址是不一样的。在自页面声明的Array所对应的构造函数是子页面的Array对象。父页面进行判断使用的Array并不等于子页面的Array.

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,588评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,456评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,146评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,387评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,481评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,510评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,522评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,296评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,745评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,039评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,202评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,901评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,538评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,165评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,415评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,081评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,085评论 2 352