手写一个getType()函数用于判断JS数据类型

简介: 前言这是一道很基础的面试题,经常用来考应届生或者实习生。但是看似如此简单的一道题,却让不少人在这上面翻了车。有些小伙伴可能觉得这无非就是考察如何判断数据类型,大家或者多少都知道一些,但是为什么会翻车呢?原因很简单:说得不全,有些概念模棱两可!今天就来总结一下如何判断数据类型,取得最优解!

1.JS 数据类型有哪些?


无非就是两大类,引用类型和值类型。


值类型:

  • String
  • Number
  • Boolean
  • Null
  • Undefined
  • Symbol


引用类型:

  • Object
  • Array
  • Function
  • RegExp
  • Date


上面的类型我相信绝大多数小伙伴都能回答上来,但是面试官让我们写的这个函数真的就判断这些类型就行了吗?


答案不是的!


面试官可能还需要让我们判断这些特殊的:NaN、BigInt、Promise、Set、Map、weakMap、Error、Infinity 等等。


2.利用 typeof 实现 getType?


不会还有人不知道 typeof 吧!这是一个非常经典的判断 js 数据类型的方法,不少小伙伴都会考虑到用 typeof 来实现我们的 getType 方法,但是我们一定要先知道 typeof 方法的特点。


typeof 特点:

  • 只能用来判断值类型(除 null),其它类型均返回 function 或者 object。
  • 对于 null 返回 object。


看表格:11.png


从上面的特点可以看出,typeof 完全不符合我们的预期!


但是,如果你非要利用 typeof 来实现我们的函数,也不是不行。我们可以结合 instanceof 来实现 getType 函数。


instanceof 简介:


instanceof 运算符用于检测构造函数的 prototype 属性是否出现在某个实例对象的原型链上。


它的主要特点是用来判断某个数据是否符合某个类型,重在判断!


使用案例:

[] instanceof Array; // true
{} instanceof Object;// true
new Date() instanceof Date;// true


从上面的代码块可以看出,instanceof 返回的是一个 Boolean 值,它可以用来判断某个数据类型是否满足我们的要求。


所以结合上面的 typeof 和 instanceof 可以实现我们的 getType 函数。


示例代码:

<script>
  function getType(data) {
    // 先判断该数据是不是基础类型
    if (typeof data === 'object') {
      // 数组类型
      if (data instanceof Array) {
        return 'array'
      } else if (data instanceof Date) {
        return 'date'
      } else if (data instanceof Map) {
        return 'map'
      }
      // 还有超多的 else...if
    } else {
      return typeof data
    }
  }
  console.info(getType([1, 2, 3])); // array
</script>

上面的代码的确可以正确返回数据类型,但是小伙伴们可能发现我 else...if 没有写完,因为是在太多了,我也写不下去了!而且还有一个很大的问题,instanceof 是无法判断 null 类型的!


那么问题出在哪里呢?


原因就出在可变因素,你根本不知道传输的数据会以何种状态出现,不按照套路出牌!


总体看来,typeof+instanceof 方法有以下问题:

  • 数据类型太多,要写超多的判断语句。
  • 不可控,如果新出了数据类型,必须增加代码判断。
  • 容易遗漏数据类型。


如果你能够写出上面的方法,面试官会觉得你掌握了数据类型这些知识,但是判断类型这一步做的还不够好。


3.Object.prototype.toString.call()


这个方法用来判断数据类型可以说是非常的完美了,它基本上能够判断出所有的数据类型,这也是我们最为推荐的一种。


官网描述:

toString() 方法返回一个表示该对象的字符串。

详细介绍:

每个对象都有一个 toString() 方法,当该对象被表示为一个文本值时,或者一个对象以预期的字符串方式引用时自动调用。默认情况下,toString() 方法被每个 Object 对象继承。如果此方法在自定义对象中未被覆盖,toString() 返回 "[object type]",其中 type 是对象的类型。


通常情况下我们都是以 data.toString()的方式调用该方法,它可以返回调用该对象的字符串。但是大家注意详细介绍中的这句话:“如果此方法在自定义对象中未被覆盖,toString() 返回 "[object type]"”

这里使用 call 是为了改变 toString()函数的内部 this 指向,当然也可以用 apply。


3.1 内部属性介绍


我们之所以能够借助 Object.prototype.toString.call()来判断数据类型,深层原因就是借助了 toString()这个方法可以返回对象内部属性这个特征。


内部属性是在 JS 中主要用来判断属性特征的,比如判断属性是否可修改、删除、是否可枚举等等,内部属性通常就以[[]]的形式来进行标识。内部属性主要包括数据属性、访问器属性、类属性等等。所有 typeof 返回为"object"的对象都有[[class]]内部属性,可以通过 Object.prototype.toString.call()获取。


Object.prototype.toString.call()返回示例:


console.info(Object.prototype.toString.call([])); // [object Array]
console.info(Object.prototype.toString.call("小猪课堂")); // [object String]
console.info(Object.prototype.toString.call({})); // [object Object]


对象常见的内部属性:

  • [[Prototype]]:对象的原型
  • [[Class]]:字符串对象的一种表现形式,可以通过 Object.prototype.toString.call()获取。
  • [[Get]]——获得属性值的方法
  • [[Put]]——设置属性值的方法
  • [[CanPut]]——检查属性是否可写
  • [[HasProperty]]——检查对象是否已经拥有该属性
  • [[Delete]]——从对象删除该属性
  • ......

了解了内部属性之后,我们就可以编写代码了。


3.2 getType函数


借助Object.prototype.toString.call()方法可以返回内部属性这个特征,我们就可以变向的获取数据的类型。

示例代码:

<script>
  function getType(data) {
    let originType = Object.prototype.toString.call(data); // 获取内部属性值
    let index = originType.indexOf(' '); // 以空格分割
    let type = originType.slice(index + 1, -1); // 截取
    return type.toLowerCase();
  }
  console.info(getType("小猪课堂")); // string
  console.info(getType(123)); // number
  console.info(getType(true)); // boolean
  console.info(getType(null)); // null
  console.info(getType(undefined)); // undefined
  console.info(getType({ name: "小猪课堂" })); // object
  console.info(getType([1,3,2])); // array
  console.info(getType(Promise.resolve())); // promise
  console.info(getType(new Set())); // set
  console.info(getType(new WeakMap())); // weakmap
  console.info(getType(new Date())); // date
  console.info(getType(() => {})); // function
  console.info(getType(new Map)); // map
  console.info(getType(BigInt(100))); // bigint
  console.info(getType(new RegExp(''))); // regexp
  console.info(getType((Symbol()))); // symbol
</script>

输出结果:112.png

从上述输出结果来看,我们这个方法基本上可以正确判断所有的类型,而且最大的好处是如果后续新增了类型,那么这个函数也是可以判断的。

总结


判断数据类型是非常简单的,难得是想得周全。很多小伙伴知道如何判断类型,但是底层的依据可能比较模糊,这就是拉开差距的地方。


想要视频学习,可以移步B站:小猪课堂



相关文章
|
16天前
|
存储 JavaScript 前端开发
JavaScript中的数据类型以及存储上的差别
通过本文的介绍,希望您能够深入理解JavaScript中的数据类型及其存储差别,并在实际编程中灵活运用这些知识,以提高代码的性能和稳定性。
44 3
|
2月前
|
JavaScript 前端开发 Java
[JS]同事:这次就算了,下班回去赶紧补补内置函数,再犯肯定被主管骂
本文介绍了JavaScript中常用的函数和方法,包括通用函数、Global对象函数以及数组相关函数。详细列出了每个函数的参数、返回值及使用说明,并提供了示例代码。文章强调了函数的学习应结合源码和实践,适合JavaScript初学者和进阶开发者参考。
43 2
[JS]同事:这次就算了,下班回去赶紧补补内置函数,再犯肯定被主管骂
|
2月前
|
存储 JavaScript 前端开发
js中的数据类型
JavaScript 中的数据类型包括五种基本类型(String、Number、Undefined、Boolean、Null)和三种引用类型(Object、Array、Function,以及ES6新增的Symbol)。基本类型直接存储值,引用类型存储的是指向实际数据的内存地址。了解它们的区别对于掌握 JavaScript 的变量赋值和函数传参至关重要。
25 1
|
2月前
|
前端开发 JavaScript 开发者
除了 Generator 函数,还有哪些 JavaScript 异步编程解决方案?
【10月更文挑战第30天】开发者可以根据具体的项目情况选择合适的方式来处理异步操作,以实现高效、可读和易于维护的代码。
|
3月前
|
JavaScript 前端开发
JavaScript 函数语法
JavaScript 函数是使用 `function` 关键词定义的代码块,可在调用时执行特定任务。函数可以无参或带参,参数用于传递值并在函数内部使用。函数调用可在事件触发时进行,如用户点击按钮。JavaScript 对大小写敏感,函数名和关键词必须严格匹配。示例中展示了如何通过不同参数调用函数以生成不同的输出。
|
3月前
|
存储 JavaScript 前端开发
JS函数提升 变量提升
【10月更文挑战第6天】函数提升和变量提升是 JavaScript 语言的重要特性,但它们也可能带来一些困惑和潜在的问题。通过深入理解和掌握它们的原理和表现,开发者可以更好地编写和维护 JavaScript 代码,避免因不了解这些机制而导致的错误和不一致。同时,不断提高对执行上下文等相关概念的认识,将有助于提升对 JavaScript 语言的整体理解和运用能力。
|
3月前
|
存储 JavaScript 前端开发
JavaScript 数据类型详解:基本类型与引用类型的区别及其检测方法
JavaScript 数据类型分为基本数据类型和引用数据类型。基本数据类型(如 string、number 等)具有不可变性,按值访问,存储在栈内存中。引用数据类型(如 Object、Array 等)存储在堆内存中,按引用访问,值是可变的。本文深入探讨了这两种数据类型的特性、存储方式、以及检测数据类型的两种常用方法——typeof 和 instanceof,帮助开发者更好地理解 JavaScript 内存模型和类型检测机制。
125 0
JavaScript 数据类型详解:基本类型与引用类型的区别及其检测方法
|
3月前
|
JavaScript 前端开发
js教程——函数
js教程——函数
52 4
|
3月前
|
存储 JavaScript 前端开发
js中函数、方法、对象的区别
js中函数、方法、对象的区别
25 2
|
3月前
|
JavaScript 前端开发
Node.js 函数
10月更文挑战第5天
26 3