news 2026/1/11 16:38:11

深入 JavaScript 原型系统:为什么 typeof Object 是 “function”,而 typeof {} 是 “object”?

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
深入 JavaScript 原型系统:为什么 typeof Object 是 “function”,而 typeof {} 是 “object”?

在学习 JavaScript 的过程中,你是否曾被以下现象困惑过?

typeof Object; // "function" typeof {}; // "object"

明明Object是“对象”的代表,为什么它自己却是个函数?
{}才是我们日常使用的“对象”,却被识别为"object"

更令人费解的是,ObjectFunction之间还存在一种看似“循环引用”的原型关系:

Object.__proto__ === Function.prototype; // true Function.__proto__ === Function.prototype; // true

本文将带你彻底理清这两个问题背后的机制,揭开 JavaScript 原型系统的神秘面纱。


一、先回答:为什么typeof Objecttypeof {}

✅ 核心原因:它们根本不是同一类东西!

表达式实际类型说明
Object内置构造函数(函数)可调用:Object()new Object()
{}普通对象字面量不可调用,是Object的实例

根据 ECMAScript 规范,typeof可调用对象(即具有[[Call]]内部方法的对象)返回"function",否则返回"object"

所以:

  • Object是一个函数typeof Object === "function"
  • {}是一个普通对象typeof {} === "object"

💡 类比理解:

  • Object是“造车的工厂”(函数)
  • {}是“工厂生产出来的一辆车”(对象)

二、深入:ObjectFunction的原型关系

要理解上述现象,必须搞懂 JavaScript 中最核心的两个内置对象:ObjectFunction

1. 基本事实

  • 所有函数(包括ObjectArrayDate)都是Function的实例。
  • 所有普通对象(包括{}[]Function.prototype)最终都继承自Object.prototype

因此:

Object instanceof Function; // true Function instanceof Function; // true ({}).__proto__ === Object.prototype; // true

2. 它们的__proto__指向哪里?

表达式说明
Object.__proto__Function.prototypeObject是函数,由Function构造
Function.__proto__Function.prototypeFunction自身也是函数
Function.prototype.__proto__Object.prototypeFunction.prototype是普通对象
Object.prototype.__proto__null原型链终点

3. 原型关系图(文字版)

Function ──.__proto__──→ Function.prototype ──.__proto__──→ Object.prototype ──.__proto__──→ null ↑ │ (instanceof) Object ──.__proto__───────────────────────────────────────┘

🔁 这看起来像“鸡生蛋、蛋生鸡”,但实际上是 JS 引擎在初始化时预定义好的闭环结构


三、验证代码(建议在控制台运行)

// 1. typeof 差异 console.log(typeof Object); // "function" console.log(typeof {}); // "object" // 2. instanceof 验证 console.log(Object instanceof Function); // true console.log(Function instanceof Function); // true // 3. __proto__ 关系 console.log(Object.__proto__ === Function.prototype); // true console.log(Function.__proto__ === Function.prototype); // true // 4. Function.prototype 是普通对象 console.log(typeof Function.prototype); // "object" console.log(Function.prototype.__proto__ === Object.prototype); // true // 5. Object.prototype 是原型链顶端 console.log(Object.prototype.__proto__ === null); // true

四、常见误区澄清

误区1:Object是最顶层对象,所以Object.__proto__应该是null

✅ 正解:Object函数,不是普通对象。所有函数的__proto__都指向Function.prototype

误区2:Function.prototype是一个函数

✅ 正解:typeof Function.prototype返回"object",它是一个内置的普通对象,仅用于被函数实例继承。

误区3:{}Object是等价的,所以 typeof 应该一样

✅ 正解:{}Object实例,就像new Array()Array的关系一样——构造函数 vs 实例


五、延伸:如何正确判断类型?

由于typeof对对象一律返回"object"(连null也是!),我们常使用:

Object.prototype.toString.call(value).slice(8, -1)

例如:

Object.prototype.toString.call([]) // "[object Array]" Object.prototype.toString.call(new Date()) // "[object Date]" Object.prototype.toString.call(/regex/) // "[object RegExp]" Object.prototype.toString.call(null) // "[object Null]" Object.prototype.toString.call(undefined) // "[object Undefined]" Object.prototype.toString.call(123) // "[object Number]" Object.prototype.toString.call("str") // "[object String]" Object.prototype.toString.call(true) // "[object Boolean]" Object.prototype.toString.call(Symbol()) // "[object Symbol]" Object.prototype.toString.call(() => {}) // "[object Function]" Object.prototype.toString.call({}) // "[object Object]"

这正是利用了Object.prototype.toString能返回内部[[Class]]标签的特性。


六、总结

问题答案
为什么typeof Object"function"因为Object是一个可调用的构造函数
为什么typeof {}"object"因为{}是一个普通对象实例
Object.__proto__指向哪?Function.prototype
Function.__proto__指向哪?Function.prototype
原型链终点是?Object.prototype.__proto__ === null

🌟记住

  • 函数也是对象,但可调用 →typeof返回"function"
  • Object是“模具”,{}是“产品”
  • 整个原型系统由ObjectFunction共同构建

参考资料

  • ECMAScript® 2024 Language Specification
  • MDN:typeof
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/1/11 13:17:23

AD导出Gerber文件:手把手教程(从零实现)

从零搞定AD导出Gerber文件:工程师实战全指南 你有没有遇到过这样的情况——辛辛苦苦画完PCB,DRC也通过了,结果发给工厂打样时却被退回:“缺G2层”“钻孔文件没生成”“阻焊开窗太大”…… 明明觉得自己“已经导出了”&#xff0c…

作者头像 李华
网站建设 2026/1/7 23:07:25

基于ARMCortex-M4F内核的MSP432MCU开发实践【3.0】

7.2.5 SPI同步操作应用举例 eUSCI模块初始化方法如下: 1)置位UCSWRST=1; 2)在UCSWRST=1的前提下,初始化所有的eUSCI寄存器; 3)通过软件清除UCSWRST; 4)通过置位UCRXIE和/或UXTXIE使能中断。 具体可参考应用实例中关于eUSCI寄存器初始化部分的程序。 【例7.2.1】…

作者头像 李华
网站建设 2026/1/9 8:31:01

告别高延迟:基于TensorRT的实时文本生成服务架构

告别高延迟:基于TensorRT的实时文本生成服务架构 在智能客服对话刚进行到第二轮,用户就因“正在思考”卡顿超过两秒而关闭页面——这并非虚构场景,而是当前大模型应用落地中最常见的体验断点。响应速度,正悄然成为决定AI产品生死的…

作者头像 李华
网站建设 2026/1/9 10:12:36

第五章:林心

第五章:林心 地球时间,第三日 15:48。 林骁、赵小雅、刘阳、张锐站在地狱之门基地主厅的金属地板上。 灰工装技术员左臂的红五星在顶灯下泛着微光:“同步舱已预热,直接进。” 他们走向西侧通道,脚步比前两日更轻&#…

作者头像 李华
网站建设 2026/1/10 12:16:40

【设计模式】A1-单例模式

👉 更多文章、资料、干货,尽在个人主页!点击头像,获取更多~ 📚 我们将深入探讨 Java 设计模式中最为基础也最为重要的一种——单例设计模式。这不仅仅是一个模式,它关乎程序的性能、资源的合理利用以及线程…

作者头像 李华