新手必看篇——JS类型判断

在JavaScript开发中,准确判断数据类型是日常编码的高频操作,也是不少初学者容易踩坑的地方。为什么typeof null"object"? instanceof 的原理是什么?有没有一种"万能"的类型判断方法?本文将从原始类型和引用类型出发,循序渐进的带你搞懂这类型判断背后的门道。

一. 先温习JS中的两大类型阵营

在开始讨论之前,我们先快速回顾一下JS中的数据类型划分:

原始类型

string,number,boolaen,null, undefined, bigint, Symbol

它们直接存储于栈中,赋值给变量时是值的拷贝。

引用类型

Array, function, object, Date

它们存储在堆内存中,变量保存的只是内存地址的引用。

因为这两种类型在底层存储和行为的差异,导致它们各自适合不同的判断方式。

二. typeof---------简单直接,但有局限

typeof 操作符是我们最常用的类型判断工具之一。它的特点是:

  • 能准确判断除null以外的所有原始类型

  • 对于引用类型,typeof 统一返回 "object",只有一个特例:函数返回 "function"

也就是说,数组、普通对象、日期对象、正则表达式等统统被识别为 "object"

js 复制代码
typeof []         // "object"
typeof {}         // "object"
typeof new Date() // "object"
typeof /reg/      // "object"
typeof function(){}  // "function"

让人迷惑的 null

js 复制代码
typeof null  // "object"

为什么判断null返回的是object呢?这是因为:typeof 是通过将值转为二进制 ,来判断类型的,二进制前三位是 0 的统一被认为是引用类型 ,在计算机中,所有的引用类型被转为二进制,前三位都是 0(除了函数) ,而 null 转为二进制是一整串 0,所以null被错误判断为了object

三.instanceof------基于原型链的归属判断

1. 只能判断引用类型,无法判断原始类型

3. 它是通过隐式原型链来查找 x 是否隶属于 X 这个类型

js 复制代码
[] instanceof Array        // true
{} instanceof Object       // true
new Date() instanceof Date  // true
function(){} instanceof Function  // true

四. Object.prototype.toString------终极万能判断法

如果你想要一个绝对可靠、能区分所有数据类型 的方法,那非Object.prototype.toString莫属。

原理浅析

每个对象内部都有一个 [[Class]] 属性,它标记了该对象的类型标签 (如 ArrayDateNumberString 等)。当我们调用 Object.prototype.toString 方法时,它会执行以下步骤:

  1. 如果值是 undefined,直接返回 "[object Undefined]"

  2. 如果值是 null,返回 "[object Null]"

  3. 否则,将值转换为一个对象(ToObject(this)),然后读取该对象的 [[Class]] 属性,最后拼接成 "[object " + [[Class]] + "]" 的形式。

由于 Object.prototype.toString 可以被任意对象调用,我们通常使用 callapply 来改变它的 this 指向,使其能直接处理原始值。

js 复制代码
Object.prototype.toString.call(123)     // "[object Number]"
Object.prototype.toString.call("hello") // "[object String]"
Object.prototype.toString.call(true)    // "[object Boolean]"
Object.prototype.toString.call(null)    // "[object Null]"
Object.prototype.toString.call(undefined) // "[object Undefined]"
Object.prototype.toString.call([])       // "[object Array]"
Object.prototype.toString.call({})       // "[object Object]"
Object.prototype.toString.call(new Date()) // "[object Date]"
Object.prototype.toString.call(/reg/)    // "[object RegExp]"
Object.prototype.toString.call(function(){}) // "[object Function]"

五. Array.isArray------专为数组而生

检测数组是最常见的需求之一,虽然Object.prototype.toString.call(arr) === "[object Array]"万无一失,但是数组有一个专门的Array.isArray()方法,更加简单直接。

js 复制代码
Array.isArray([])      // true
Array.isArray({})      // false
Array.isArray(new Array(5))  // true

六.各方法对比总结

方法 适用场景 不足之处
typeof 判断原始类型(除null外)和函数 typeof null"object";无法区分具体的对象类型(数组、日期等)
instanceof 判断某个对象是否由指定构造函数创建 无法判断原始类型
Object.prototype.toString 全场景通用,精准返回每个数据的内置类型 写法稍长,通常需要封装
Array.isArray 专用于判断数组 只解决数组场景
相关推荐
小妖6661 小时前
console.log 显示内容不全怎么办
javascript·js·console.log
小小高不懂写代码1 小时前
Vibe Coding时代的自我鞭策
前端·人工智能
喵个咪1 小时前
基于 Nuxt 4 的现代 Headless CMS 前端:架构深度解析与二次开发指南
前端·vue.js·nuxt.js
AI科技星1 小时前
万有引力G与真空介电常数ε0全维度完整关系式汇编(基于v=c螺旋时空理论)
c语言·开发语言·前端·javascript·网络·汇编·electron
didadida2621 小时前
第二回: Session Assistant 工具链的三节点设计
javascript·agent
云间寄信2 小时前
异步编程与事件循环
javascript
喵个咪2 小时前
基于 Next.js 的 Headless CMS 前端架构:技术解析与二次开发导引
前端·react.js·next.js
阿白同学1054512 小时前
一座前端文明的地层:React 源码考古报告
前端
七牛云行业应用2 小时前
别手搓多Agent了!Codex Windows版用Git Worktree并行跑代码,真的香
前端