typeof和instanceof原理

目录

  • JavaScript数据类型
  • typeof
  • JavaScript原型链
  • instanceof

JavaScript数据类型

JavaScript有八种内置类型

  • 空值(null)
  • 未定义(undefined)
  • 布尔值(boolean)
  • 数字(number)
  • 字符串(string)
  • 对象 (object)
  • 符号(symbol, ES6中新增)
  • 大整数(BigInt, ES2020 引入)

除对象外,其他统称为“基本类型”。

typeof null // 'object'
typeof undefined; // "undefined"
typeof false; // "boolean"
typeof 1; // "number"
typeof '1'; // "string"
typeof {}; // "object" 
typeof []; // "object" 
typeof new Date(); // "object"

typeof Symbol; // "Symbol"
typeof 123n // 'bigint'

这里的类型值的是值,变量是没有类型的,变量可以随时持有任何类型的值。JavaScript中变量是“弱类型”的,一个变量可以现在被赋值为 字符串类型,随后又被赋值为数字类型。

typeof是一个操作符而不是函数,用来检测给定变量的数据类型。

Symbol 是ES6中引入的一种原始数据类型,表示独一无二的值。BigInt(大整数)是 ES2020 引入的一种新的数据类型,用来解决 JavaScript中数字只能到 53 个二进制位(JavaScript 所有数字都保存成 64 位浮点数,大于这个范围的整数,无法精确表示的问题。(在平常的开发中,数据的id 一般用 string 表示的原因)。为了与 Number 类型区别,BigInt 类型的数据必须添加后缀n。 1234为普通整数,1234nBigInt。了解更多可以看 《ES6 入门教程》

typeof null 为什么返回 'object',稍后会从JavaScript数据底层存储机制来解释。

还有一种情况

function foo() {};
typeof foo; // 'function'

这样看来,function 也是JavaScript的一个内置类型。然而查阅规范,就会知道,它实际上是 object 的一个"子类型"。具体来说,函数是“可调用对象”,它有一个内部属性[[call]],该属性使其可以被调用。typeof 可以用来区分函数其他对象。

但是使用 typeof不能 判断对象具体是哪种类型。所有typeof 返回值为 "object" 的对象(如数组,正则等)都包含一个内部属性 [[class]](我们可以把它看做一个内部的分类)。这个属性无法直接访问,一般通过 Object.prototype.toString(...)来查看。

Object.prototype.toString.call(new Date); // "[object Date]"
Object.prototype.toString.call([]); // "[object Array]"
Object.prototype.toString.call(/reg/ig); // "[object RegExp]"

instanceof 运算符也常常用来判断对象类型。用法: 左边的运算数是一个object,右边运算数是对象类的名字或者构造函数; 返回truefalse

[] instanceof Array; // true
[] instanceof Object; // true
[] instanceof RegExp; // false
new Date instanceof Date; // true

instanceof 的内部机制是:检测构造函数的 prototype 属性是否出现在某个实例对象的原型链上。下面会详解介绍该部分。

typeof 原理

typeof原理: 不同的对象在底层都表示为二进制,在Javascript中二进制前(低)三位存储其类型信息

  • 000: 对象
  • 010: 浮点数
  • 100:字符串
  • 110: 布尔
  • 1: 整数

typeof null 为"object", 原因是因为 不同的对象在底层都表示为二进制,在Javascript中二进制前(低)三位都为0的话会被判断为Object类型,null的二进制表示全为0,自然前三位也是0,所以执行typeof时会返回"object"。
一个不恰当的例子,假设所有的Javascript对象都是16位的,也就是有16个0或1组成的序列,猜想如下:

Array: 1000100010001000
null:  0000000000000000

typeof []  // "object"
typeof null // "object"

因为Array和null的前三位都是000。为什么Array的前三位不是100?因为二进制中的“前”一般代表低位, 比如二进制00000011对应十进制数是3,它的前三位是011。

instanceof

要想从根本上理解,需要从两个方面入手:

  • 语言规范中是如何定义这个运算符的
  • JavaScript原型继承机制

通俗一些讲,instanceof 用来比较一个对象是否为某一个构造函数的实例。注意,instanceof运算符只能用于对象,不适用原始类型的值。

  1. 判断某个实例是否属于某种类型
function Foo() {};
Foo.prototype.message = ...;
const a = new Foo();
  1. 也可以判断一个实例是否是其父类型或者祖先类型的实例。
function Car(make, model, year) {
  this.make = make;
  this.model = model;
  this.year = year;
}
const auto = new Car('Honda', 'Accord', 1998);

console.log(auto instanceof Car);
// expected output: true

console.log(auto instanceof Object);
// expected output: true

JavaScript原型链

理解原型

我们创建的每个函数都有一个 [prototype])属性,这个属性是一个指针,指向一个对象,而这个对象的用途是包含可以由特定类型的所有实例共享的属性和方法。那么 prototype 就是调用 构造函数 而创建的那个对象实例的原型对象。使用原型对象的好处是可以让所有对象实例共享它所包含的属性和方法。

function Person() {};
Person.prototype.name = 'kangkang';
Person.prototype.sayName = function() {
    console.log(this.name);
}

const person1 = new Person();
person1.sayName(); // 'kangkang'

const person2 = new Person();
person2.sayName(); // 'kangkang'

console.log(person1.sayName === person2.sayName);
// true

构造函数,原型和实例的关系

  • 每个构造函数都有一个原型对象
  • 原型对象都包含一个指向构造函数指针
  • 实例都包含一个指向原型对象指针

那么,假如我们让原型对象等于另一个类型实例,结果会怎么样?
显然,此时的原型对象将包含一个指向另一个原型指针,相应地,另一个原型中也包含着一个指向指向另一个构造函数指针。假如另一个原型又是另一个类型实例,那么上述关系依然成立,如此层层递进,就构成了实例与原型的链条。这就是所谓原型链的基本概念。

上面这段话有点绕,如果想不明白的话,这里可以停一下,读三篇,再结合我们平常写代码使用过程中的实际场景。

[[prototype]]机制

[[prototype]]机制就是存在与对象中的一个内部链接,它会引用其他对象。
通常来说,这个链接的作用是:如果在对象上没有找到需要的属性或者方法引用,引擎就会继续在 [[ptototype]]关联的对象上进行查找,同理,如果在后者中也没有找到需要的引用就会继续查找它的[[prototype]],以此类推。这一系列对象的链接被称为“原型链”。

但是哪里是 [[prototype]]的 ”尽头“呢?

所有普通的 [[prototype]]链最终都会执行内置的 Object.prototype。由于所有的"普通"(内置,不是特定主机的扩展)对象都”源于“(或者说把[[prototype]] 链顶端设置为)这个Object.prototype对象,所以说它包含JavaScript中许多通用的功能。比如说.toString().valueOf()等等

Object.prototype是js原型链的最顶端,它的__proto__null(有proto属性,但值是 null,因为这是原型链的最顶端);

为什么要这么设计?

最主要的就是节省内存,如果属性和方法定义在原型上,那么所有的实例对象就能共享。

__proto__

绝大多数(不是所有)浏览器也支持一种非标准的方法来访问内部的 [[prototype]]属性。

function Foo() {};
const a = new Foo();

a.__proto__ === Foo.prototype; // true

这个奇怪的.__proto__属性“神奇地”引用了内部的[[prototype]]对象。如果你想直接查找(甚至可以直接通过.proto.proto ...来遍历)原型链的话,这个方法非常有用。

.construtor一样,__proto__实际上并不存在于你正在使用的对象(本例中是a)。实际上,它和其他的常用函数(.toString()、.isPrototypeOf(...),等等 一样,存在于内置的Object.prototype中。(它们是不可枚举的;

此外,.__proto__看起来很像一个属性,但是实际上它更像一个 getter/setter
.__proto__的实现大致是这样的

Object.defineProperty(Object.prototype, "__proto__", {
    get: function() {
        return Object.getPrototypeOf(this);
    },
    // ES6中的Object.setPrototypeOf
    set: function(o) {
        Object.setPrototypeOf(this, o);
        return o;
    }
})

因此,访问(获取值) a.__proto__时,实际上是调用了 a.__proto__()(调用getter函数)。虽然getter函数存在于Object.prototype对象中,但是 它的 this 指向对象a,所以和object.getPrototypeOf(a)结果相同。

.__proto__是可设置属性,之前的代码中使用ES6的Object.setPrototypeOf(...)进行设置。然而,通常来说你不需要修改已有对象的[[prototype]]

原型链

JavaScript原型链
    1. function Foo 就是一个方法,比如内置的 Array,String,或者自定义方法。
    1. function Object就是 Object
    1. function Function就是 Function
    1. 以上三个其实都是 function,所以他们的 __proto__都是 Function.prototype
    1. 记住 String, Array, Number,Object, Function这些其实都是 function
function Foo() {};

console.log(Object instanceof Object); // true
console.log(Function instanceof Function); // true
console.log(Function instanceof Object); // true

console.log(Foo instanceof Foo); // false
console.log(Foo instanceof Object); // true
console.log(Foo instanceof Function); // true

大家可以在控制台输出,可以直观的看到每个步骤的输出,结合instanceof 的规范跟js原型链 加深理解。

回过头来再看instanceof

instanceof的语法:

object instanceof constructor
// 等同于
constructor.prototype.isPrototypeOf(object)
  • object: 要检测的对象
  • constructor:某个构造函数

instanceof的代码实现。

function instanceof(L, R) { //L是表达式左边,R是表达式右边
    const O = R.prototype;
    L = L.__proto__;
    while(true) {
        if (L === null)
            return false;
        if (L === O) // 这里重点:当 L 严格等于 0 时,返回 true 
            return true;
        L = L.__proto__;
    }
}

instanceof原理: 检测 constructor.prototype是否存在于参数 object的 原型链上。instanceof 查找的过程中会遍历object的原型链,直到找到 constructorprototype ,如果查找失败,则会返回false,告诉我们,object 并非是 constructor 的实例。

原型链这部分很不好理解,我基本上都是看完过几天就忘,所以要多看几遍多理解,花些时间搞明白,搞明白这部分。之后再看相关的东西,就很简单易懂。这部分是JavaScript很重要的核心。花几天时间反复看,弄明白了,以后理解很多问题都是简单的多。如果你发现我上面哪部分表述的不太准确,记得给我指出来,互相学习。这部分推荐好好看看 《JavaScript高级程序设计(第3版)》第六章的这部分,还有 《你不知道的JavaScript(上卷)》第五章关于这部分内容的讲解。

Symbol.hasInstance

对象的Symbol.hasInstance属性,指向一个内部方法。当其他对象使用instanceof运算符,判断是否为该对象的实例时,会调用这个方法。比如,foo instanceof Foo在语言内部,实际调用的是Foo[Symbol.hasInstance](foo)

class MyClass {
  [Symbol.hasInstance](foo) {
    return foo instanceof Array;
  }
}

[1, 2, 3] instanceof new MyClass() // true

总结

看完之后,脑子里可以把上面的内容串一下;看看下面的几个问题你是否可以立刻想出来

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

推荐阅读更多精彩内容