【前端】理解 JavaScript 中 typeof 操作符的独特行为


在这里插入图片描述

博客主页: [小ᶻ☡꙳ᵃⁱᵍᶜ꙳]
本文专栏: 前端

文章目录

  • 💯前言
  • 💯`typeof` 操作符的基本使用
  • 💯为什么 `typeof` 数组是 "object"?
  • 💯为什么 `typeof {}` 返回 "object"?
  • 💯为什么 `typeof null` 也是 "object"?
  • 💯`typeof null` 返回 "object" 的历史原因
  • 💯其它 `typeof` 的怪异行为
  • 💯更好的类型判断方式
  • 💯最佳实践与建议
  • 💯小结


在这里插入图片描述


💯前言

  • JavaScript 是一种灵活且动态的编程语言,其设计中包含了诸多独特之处。这些特性为开发者提供了极大的自由度,但也伴随着某些复杂性,尤其是在类型系统的处理上。typeof 操作符便是其中一个常见的例子,它的表现往往让新手乃至经验丰富的开发者感到困惑。在本文中,我们将系统性地探讨typeof操作符的行为,特别是在处理数组对象null时的表现,并揭示这些行为背后的深层原因,以及如何在实践中应对这些情况。为了全面理解这些问题,我们将深入分析这些类型的历史背景,并提供实践中可以应用的一些实用技巧
    JavaScript在这里插入图片描述


💯typeof 操作符的基本使用

在这里插入图片描述
在 JavaScript 中,typeof 是一个用于检测变量类型的操作符。它返回一个字符串,表示变量的数据类型。最常见的类型包括:

  • "number"
  • "string"
  • "boolean"
  • "object"
  • "undefined"
  • "function"

在日常开发过程中,typeof 操作符通常是一个非常方便的工具,用于快速判断变量的类型。然而,JavaScript 中的一些类型判断行为可能会让人感到困惑,特别是在涉及数组、普通对象以及 null 的时候。这种困惑源于 JavaScript 语言设计中的某些妥协和历史遗留问题,它们塑造了当前语言中某些行为的独特性和不一致性。


💯为什么 typeof 数组是 “object”?

首先让我们来讨论一个很常见的例子,即对数组的类型判断:

let arr = [1, 2, 3];
console.log(typeof arr); // 输出 "object"

在这里插入图片描述

在 JavaScript 中,数组实际上是对象的一种特殊类型。虽然数组和普通对象有显著的不同,例如数组具有 length 属性,提供了诸如 pushpop 等特殊的方法,并且其结构通常是有序的,但在底层,JavaScript 实现数组时仍将其视为对象。因此,typeof 操作符会返回 "object",这可能与直觉不符。

为了更加精确地判断一个变量是否是数组,我们可以使用 Array.isArray() 方法:

console.log(Array.isArray(arr)); // 输出 true

在这里插入图片描述

Array.isArray() 是判断数组的最佳方式,因为它能够准确地辨别一个值是否为数组,而不依赖于 typeof 的笼统返回值。之所以设计这个方法,是因为开发者通常希望将数组与普通对象区别开来,而 typeof 由于其固有的局限性无法做到这一点。


💯为什么 typeof {} 返回 “object”?

普通对象是 JavaScript 中的基本构造块之一。它们通过字面量 {} 来创建,可以用来存储键值对:

let obj = {};
console.log(typeof obj); // 输出 "object"

在这里插入图片描述

对于字面量 {}typeof 返回 "object" 是符合直觉的,因为 {} 代表的是 JavaScript 中典型的对象类型。对象是 JavaScript 中的一种复合数据类型,其主要功能是表示键值对的集合,且允许动态地添加或移除属性。理解对象在 JavaScript 中的表现形式以及如何被 typeof 识别,对掌握这一语言至关重要,尤其是对象几乎参与了所有复杂数据结构的构建。


💯为什么 typeof null 也是 “object”?

接下来,我们来讨论一个颇具迷惑性的问题:

console.log(typeof null); // 输出 "object"

在这里插入图片描述

null 表示的是“空值”或“空引用”,通常用于指示某个对象或值为空。在绝大多数编程语言中,我们可能期望 null 是一种单独的数据类型。然而,在 JavaScript 中,typeof null 返回 "object",这显然不符合直觉,也让开发者在编写代码时容易出现困惑。


💯typeof null 返回 “object” 的历史原因

typeof null 返回 "object" 是 JavaScript 语言中的一个历史遗留问题。这种行为可以追溯到 JavaScript 最早的实现。在初期的实现中,JavaScript 采用了一种称为标签存储的方式来表示不同的数据类型。对象的标签值为 0,而 null 被视为对象的一个特殊标志,也因此被分配了与对象相同的标签值。这一实现细节导致了 typeof null 返回 "object" 的现象。

尽管这一实现方式在现代 JavaScript 中已经显得过时且不合逻辑,但由于语言的向后兼容性,无法对其进行根本性的更改,因为这种变动可能会导致大量现存代码失效。因此,这个问题一直被保留下来,成为开发者在使用 JavaScript 时必须了解的一个特性
为了避免这种困惑,开发者在判断一个值是否为 null 时,通常使用严格的相等比较运算符:

let value = null;
console.log(value === null); // 输出 true

在这里插入图片描述

通过这种方式,可以确保 valuenull,而不会被误判为其他类型的对象。这种做法不仅直观,而且可以减少由于误判类型带来的潜在错误,尤其是在需要处理复杂对象和空引用时显得尤为重要。


💯其它 typeof 的怪异行为

JavaScript 中的 typeof 操作符还存在一些其他特殊情况。例如,当 typeof 用于函数时,它会返回 "function",这一点相对合理,因为函数在 JavaScript 中是一等公民,既可以作为变量被赋值,也可以作为参数传递给其他函数,甚至可以作为返回值。

function example() {}
console.log(typeof example); // 输出 "function"

在这里插入图片描述

虽然函数在底层也是对象的一种特殊类型,但 typeof 将它们单独识别为 "function",使得它们与普通对象区分开来,从而为开发者提供了一种快速判断的方式。

另一方面,typeof undefined 返回 "undefined",这相对简单直观。它表示一个变量尚未被赋值,这种行为在变量初始化和检查时非常有用,能够快速地判断某个变量是否处于未定义状态。


💯更好的类型判断方式

由于 typeof 存在一些不直观的行为,因此在复杂应用中,我们需要更加可靠的方式来进行类型判断。以下是一些常用的类型判断方式:

  1. Array.isArray() 用于数组

    • Array.isArray(value) 是判断数组的最佳方式,因为它专门用于判断数组,无论数据结构多复杂都能准确识别。
  2. instanceof 操作符

    • instanceof 可以用来判断某个值是否是某个构造函数的实例。这种方法适用于多种不同类型的数据结构,尤其是在涉及自定义类和继承时。
    console.log([] instanceof Array); // 输出 true
    console.log({} instanceof Object); // 输出 true
    

    在这里插入图片描述

  3. Object.prototype.toString.call()

    • Object.prototype.toString.call(value) 是一种非常强大的类型判断方法,能够精确返回值的类型信息:
    console.log(Object.prototype.toString.call([])); // 输出 "[object Array]"
    console.log(Object.prototype.toString.call({})); // 输出 "[object Object]"
    console.log(Object.prototype.toString.call(null)); // 输出 "[object Null]"
    

    在这里插入图片描述

    这种方式能够提供更加详细的类型信息,尤其是在处理复杂和不确定的数据时非常有用。它被视为所有类型判断方法中最通用的方式,特别适合于需要对输入进行全面验证的应用场景。


💯最佳实践与建议

  1. 避免直接用 typeof 判断数组和 null

    • 对于数组,应使用 Array.isArray() 进行判断。
    • 对于 null,直接使用 === null 进行判断。
  2. 理解并适应历史遗留问题

    • 理解 JavaScript 中 typeof 的行为历史原因,有助于避免编码中的常见陷阱。了解 typeof null 返回 "object" 是一个历史遗留问题,可以帮助你更快地在调试过程中发现问题。
  3. 结合使用多种类型判断方式

    • 在复杂应用中,单靠 typeof 通常不足以满足所有需求。结合使用 instanceofObject.prototype.toString.call() 等方法,可以实现更加精确和可靠的类型判断。这些方法在大型应用程序中尤为重要,特别是在处理多样化的数据时。
  4. 团队开发中的一致性

    • 在团队开发中,保持类型检查方法的一致性非常重要。例如,统一使用 Object.prototype.toString.call() 作为标准类型检查方式,可以确保所有团队成员编写的代码在风格和行为上保持一致,从而减少不必要的错误和维护成本。

在这里插入图片描述


💯小结

  • 在这里插入图片描述
    JavaScript 是一种极为灵活的语言,其设计之初为了实现简便易用做出了许多权衡,但也因此引入了一些复杂性,特别是在类型系统上。其中,typeof 操作符的行为便是一个经典的例子。通过深入理解typeof在不同场景下的表现以及其背后的历史原因,开发者可以有效避免一些常见的陷阱,并编写出更加健壮和可靠的代码。
    尽管 JavaScript 的某些设计决策显得不那么严谨,但其灵活性广泛的生态系统使得它在现代 Web 应用开发中占据了重要地位。理解并掌握这些细微的特性不仅能够帮助开发者编写更具稳定性的代码,还能在调试和代码审查过程中显著提高效率

在这里插入图片描述


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/483035.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Github提交Pull Request教程 Git基础扫盲(零基础易懂)

1 PR是什么? PR,全称Pull Request(拉取请求),是一种非常重要的协作机制,它是 Git 和 GitHub 等代码托管平台中常见的功能,被广泛用于参与社区贡献,从而促进项目的发展。 PR的整个过…

C/C++ 数据结构与算法【线性表】 顺序表+链表详细解析【日常学习,考研必备】带图+详细代码

1)线性表的定义 线性表(List):零个或多个数据元素的有限序列。 线性表的数据集合为{a1,a2,…,an},假设每个元素的类型均为DataType。其中,除第一个元素a1外,每一个元素有且只有一个直接前驱元素…

浏览器的数据六种存储方法比较 :LocalStorage vs. IndexedDB vs. Cookies vs. OPFS vs. WASM-SQLite

在构建该 Web 应用程序,并且希望将数据存储在用户浏览器中。也许您只需要存储一些小标志,或者甚至需要一个成熟的数据库。 我们构建的 Web 应用程序类型发生了显着变化。在网络发展的早期,我们提供静态 html 文件。然后我们提供动态渲染的 h…

【C++boost::asio网络编程】有关异步读写api的笔记

异步读写api 异步写操作async_write_someasync_send 异步读操作async_read_someasync_receive 定义一个Session类&#xff0c;主要是为了服务端专门为客户端服务创建的管理类 class Session { public:Session(std::shared_ptr<asio::ip::tcp::socket> socket);void Conn…

芯片测试-RF中的S参数,return loss, VSWR,反射系数,插入损耗,隔离度等

RF中的S参数&#xff0c;return loss, VSWR&#xff0c;反射系数&#xff0c;插入损耗&#xff0c;隔离度 &#x1f4a2;S参数&#x1f4a2;&#x1f4a2;S11与return loss&#xff0c;VSWR&#xff0c;反射系数&#x1f4a2;&#x1f4a2;S21&#xff0c;插入损耗和增益&#…

2024年Java面试八股文大全(附答案版)

很多人会问Java面试八股文有必要背吗&#xff1f; 我的回答是&#xff1a;很有必要。你可以讨厌这模式&#xff0c;但你一定要去背&#xff0c;因为不背你就进不了大厂。 国内的互联网面试&#xff0c;恐怕是现存的、最接近科举考试的制度。 而且&#xff0c;我国的八股文确…

我不是挂王-用python实现燕双鹰小游戏

一.准备工作 1.前言提要 作为程序员在浩瀚的数字宇宙中&#xff0c;常常感觉现实世界是一台精密运作的虚拟机&#xff0c;其底层的物理逻辑如同铁律般难以撼动。然而我们拥有在虚拟世界中自由驰骋、创造无限可能的独特力量。突发奇我想用Python写出燕双鹰的小游戏,这样想想就很…

Python语法基础(七)

&#x1f308;个人主页&#xff1a;羽晨同学 &#x1f4ab;个人格言:“成为自己未来的主人~” 我们今天的这篇文章&#xff0c;主要和大家讲一下函数重写的问题。 首先&#xff0c;我们要知道的是&#xff0c;函数重写是有前提条件的&#xff0c;那就是继承。 自定义函数…

NaviveUI框架的使用 ——安装与引入(图标安装与引入)

文章目录 概述安装直接引入引入图标样式库 概述 &#x1f349;Naive UI 是一个轻量、现代化且易于使用的 Vue 3 UI 组件库&#xff0c;它提供了一组简洁、易用且功能强大的组件&#xff0c;旨在为开发者提供更高效的开发体验&#xff0c;特别是对于构建现代化的 web 应用程序。…

神经网络入门实战:(九)分类问题 → 神经网络模型搭建模版和训练四步曲

(一) 神经网络模型搭建官方文档 每一层基本都有权重和偏置&#xff0c;可以仔细看官方文档。 pytorch 官网的库&#xff1a;torch.nn — PyTorch 2.5 documentation Containers库&#xff1a;用来搭建神经网络框架&#xff08;包含所有的神经网络的框架&#xff09;&#xff1b…

以AI算力助推转型升级,暴雨亮相CCF中国存储大会

2024年11月29日-12月1日&#xff0c;CCF中国存储大会&#xff08;CCF ChinaStorage 2024&#xff09;在广州市长隆国际会展中心召开。本次会议以“存力、算力、智力”为主题&#xff0c;由中国计算机学会&#xff08;CCF&#xff09;主办&#xff0c;中山大学计算机学院、CCF信…

Java中的“抽象类“详解

1.抽象类的定义 在面向对象的概念中,所有的对象都是通过,类来描述的,但是反过来,不是所有的类都是用来描述对象的,如果一个类中没有包含足够的信息来描述一个具体的对象,这样的类就是抽象类 抽象类的概念是比较抽象的,比如现在有一个"父类"是"动物类",继…

NVR监测软件EasyNVR多个NVR同时管理:录播主机的5条常见问题与解决办法

视频监控广泛应用于城市治安、交通管理、商业安保及家庭监控等领域。在使用EasyNVR平台管理多个NVR设备时&#xff0c;尤其是涉及到海康录播主机的场景中&#xff0c;使用者可能会遇到一些常见问题。本文将探讨海康录播主机的五个常见问题及其解决办法。 1、海康录播主机的5条常…

多级缓存设计实践

缓存是什么&#xff1f; 缓存技术是一种用于加速数据访问的优化策略。它通过将频繁访问的数据存储在高速存储介质&#xff08;如内存&#xff09;中&#xff0c;减少对慢速存储设备&#xff08;如硬盘或远程服务器&#xff09;的访问次数&#xff0c;从而提升系统的响应速度和…

Linux网络编程之---多线程实现并发服务器

下面我们来使用tcp集合多线程实现并发服务器 一.服务端 #include <stdio.h> #include <arpa/inet.h> #include <unistd.h> #include <stdlib.h> #include <string.h> #include <pthread.h>typedef struct sockinfo {char ip[16];unsigne…

Web API基本认知

作用和分类 作用&#xff1a;就是使用JS去操作html和浏览器 分类&#xff1a;DOM&#xff08;文档对象模型&#xff09;、BOM&#xff08;浏览器对象模型&#xff09; 什么是DOM DOM&#xff08;Document Object Model ——文档对象模型&#xff09;是用来呈现以及与任意 HTM…

《Python基础》之Numpy库

目录 简介 一、创建数组 1、根据列表创建数组 2、创建全0数组 3、创建全1数组 4、创建单位矩阵 5、创建随机数数组 二、查看数组的属性 三、 数组的操作 1、索引和切片 2、变形 3、拼接 &#xff08;1&#xff09;、vstack() 纵向拼接 &#xff08;2&#xff09;、hs…

人工智能-卷积神经网络(学习向)

一.概述&#xff1b; 卷积神经网络&#xff08;Convolutional Neural Network, CNN&#xff09;是一种专门用于处理具有类似网格结构的数据&#xff08;如图像&#xff09;的深度学习模型。 主要用于处理机器视觉任务。 主要功能&#xff1b; 1.图像分类 2.目标检测 3.图像分割…

思维导图+实现一个登录窗口界面

QQ2024122-205851 import sys from PyQt6.QtGui import QIcon, QPixmap, QMovie from PyQt6.QtWidgets import QApplication, QWidget, QLineEdit, QPushButton, QLabel, QVBoxLayout# 封装我的窗口类 class LoginWidget(QWidget):# 构造函数def __init__(self):# 初始化父类su…

使用 Pytorch 构建 Vanilla GAN

文章目录 一、说明二、什么是 GAN&#xff1f;三、使用 PyTorch 的简单 GAN&#xff08;完整解释的代码示例&#xff09;3.1 配置变量3.2 、PyTorch 加速3.3 构建生成器3.4 构建鉴别器 四、准备数据集五、初始化函数六、前向和后向传递七、执行训练步骤八、结果 一、说明 使用…