为什么用 void 0 代替 undefined?

简介: 为什么用 void 0 代替 undefined?

前言


众所周知,void 运算符总会返回一个 undefined 的结果。


那么,为什么要用 void 0 代替 undefined 呢?这不是多此一举了吗?


不急,慢慢道来~


正文


一、void 运算符


语法非常简单:


void expression


它将会对表达式进行求值,然后返回 undefined 的「原始值」。那么,void 0 不就是得到 undefined 最简单的写法嘛!(请注意,写法上 void 0 相当于 void(0))。


我们知道,立即函数执行表达式(IIFE)要非常多的写法,使用 void 关键字也是可以的,比如:


void (function () {
  // some statements
})()
// 相当于
void function () { /* some statements */ }()


另外,你应该看过以下类似的写法:


<a href="javascript:0;">Link</a>


我们知道 javascript: 是 JavaScript 中的伪协议,表示将会使用 JS 解析器执行其后的所有语句,若其最后一个JavaScript 语句返回值不为 undefined,那么其返回值将会替换为页面内容。


就以上 HTML 标签,在不同浏览器下点击,会有不同的结果。其中 Chrome、Safari 中点击无任何反馈,而 Firefox 中页面内容将会被替换为 0


为了解决以上差异表现,通常的做法就是使用 void 关键字,比如:


<a href="javascript:void(0);">Link</a>


这也是处理 <a> 标签默认行为的方式之一。


这种伪协议除了在事件处理程序中使用,也可在浏览器地址栏、书签地址中使用。

除此之外,还有一种常用的用法:当箭头函数中只有一行语句,且不需要返回值时,则可以:

element.onclick = () => void doSomething()


二、undefined


你有可能不知道,我们天天写的 undefined,它其实是一个「全局对象」中的一个「属性」。

// 浏览器宿主环境
window.undefined === undefined // true
// Node 宿主环境
global.undefined === undefined // true


除了 undefined 之外,类似的还有 NaNInfinity 也是全局对象的属性。它们的属性描述对象都是:不可写、不可枚举、不可配置(详看 ECMAScript 19.1 Value Properties of the Global Object)。

{
  [[writable]]: false,
  [[enumerable]]: false
  [[configurable]]: false
}


当然,这个在 ES5 之前是可以被改写的,下面可以对比一下:


53.webp.jpg

IE 8

54.webp.jpg

Chrome 100


从这个角度看是不是可以理解为什么用 void 0 而不是 undefined 这个全局属性了。

还有,需要特别注意的是:


undefined 并不是 ECMAScript 标准中的一个保留字,因此它是可以被作为变量标识符而使用的(但项目中千万别这么用),就行 window 一样。


在 ECMAScript 标准中,保留字有以下这些(详见 12.6.2 Keywords and Reserved Words):


56.webp.jpg


在现代浏览器下,请对比以下三种示例:

// 示例一
!(function () {
  var undefined = 1
  console.log(undefined) // 结果是?
})()

// 示例二
!(function () {
  undefined = 2
  console.log(undefined) // 结果是?
})()

// 示例三
var undefined = 3
console.log(undefined) // 结果是?

// 示例四
let undefined = 4
console.log(undefined) // 结果是?


前三个打印结果分别是:1undefinedundefined,最后一个则在 let undefined = 4 处抛出 SyntaxError。


原因分析:


  • 示例一的 undefined 被声明为一个变量标识符,并赋值为 1,因此打印结果为 1
  • 示例二中 undefined = 2 表示修改全局变量的 undefined 属性(window.undefined),由于它是 non-writable 的,因此不会被重写,打印结果还是其默认值 undefined
  • 示例三与示例二同理,只是它们所处的作用域不同罢了;
  • 示例四原意是为了说明 varlet/const 的区别,在全局作用域下,前者声明的变量会被添加至全局对象上,而后者则不会。但是实际执行下来,会抛出语法:Uncaught SyntaxError: Identifier 'undefined' has already been declared(目测是默认情况下就有一个 var undefined 的声明的语句了,因此使用 let undefined 重复声明会抛出错误,这种猜测暂未在标准中找到佐证)。

不知道有没有人会分不清:undefined 什么时候作为原始值?什么时候是作为变量(属性)?

var foo // 此处 foo 的值是原始值 undefined;
function bar() {}
bar() // 此处函数 bar 的返回值是原始值 undefined;
void 0 // 此处 void 操作的返回值是原始值 undefined;
foo == undefined // 此处相等运算符右侧的 undefined 是全局对象的 undefined 属性。

到这里,你应该对于 undefined 有了一个比较全面的认识了。


我猜你应该看过类似以下的「小心机」代码:

;(function ($, window, undefined) {
  // some statements
})(jQuery, window)


如果在 IIFE 中使用 undefined,它其实是引用了形参 undefined。但形参 undefined 的值就是 undefined 的原始值,因为在调用函数时并未传入实参。除了能确保 undefined 的值是原始值之外,还能加速变量 undefined 的查找,由于它在函数作用域内就能找到该变量,就不会继续往作用域链上查找。


三、用 void 0 代替 undefined?


void 0 总是返回原始值 undefined,无论全局属性 undefined 是否被改写,它都能确保其值是 undefined(原始值)。


比如著名的工具库 underscore 大量使用了 void 0 来代替 undefined,再者 UglifyJS、Terser 等代码压缩工具也会将 undefined 转换为 void 0,这样可以节省一些字节:

function isUndefined(x) {
 return x === undefined
}
// Minified
function isUndefined(n){return void 0===n}


但我们在编写代码的时候,直接使用 undefined 也是没有太大问题的,注意下前面提到的一些点就好了,其余的就交由工具来处理即可。

目录
相关文章
|
JavaScript 前端开发 安全
TypeScripe笔记:any、unknown、never、void、null 和 undefined 及其比较
TypeScripe 中,any、unknown、never、void、null 和 undefined 的比较
177 0
|
JavaScript 安全
Typescript any、unknown、 void、undefined 和 never 类型
本文介绍了 Typescript 的 `any` 类型和 `unkonw` 类型,它们都简化了在 TS 中对类型的使用,方便快速开发,相比于 `any`,`unkonw` 类型更加安全。 然后介绍了 Typescript 的几个特殊的类型,`void`,`undefined`,`never`,它们主要应用在几个特定的场景,比较容易区分。
770 0
|
7月前
|
前端开发 小程序 JavaScript
微信小程序-Unhandled promise rejection TypeError: Cannot read property ‘get‘ of undefined
微信小程序-Unhandled promise rejection TypeError: Cannot read property ‘get‘ of undefined
|
5月前
|
存储 JavaScript 前端开发
成功解决:Cannot read properties of undefined (reading ‘commit‘)
这篇文章提供了解决Vuex中"Cannot read properties of undefined (reading 'commit')"错误的两种方法:检查模板中的数据属性是否存在,以及确保在Vue实例中正确挂载了store对象。
成功解决:Cannot read properties of undefined (reading ‘commit‘)
|
5月前
|
定位技术 Apache
Echarts——Invalid geoJson format Cannot read property 'length' of undefined
Echarts——Invalid geoJson format Cannot read property 'length' of undefined
117 0
|
5月前
|
JavaScript
VUE——filemanager-webpack-plugin报错TypeError: Cannot read property 'isFile' of undefined
VUE——filemanager-webpack-plugin报错TypeError: Cannot read property 'isFile' of undefined
104 0
|
5月前
|
前端开发 JavaScript
VUE——Uncaught (in promise) TypeError: Cannot read property '__esModule' of undefined
VUE——Uncaught (in promise) TypeError: Cannot read property '__esModule' of undefined
116 0
TypeError: Cannot set properties of undefined (setting ‘resdata‘),res定义数据出现的问题,定义的方法用this换成that
TypeError: Cannot set properties of undefined (setting ‘resdata‘),res定义数据出现的问题,定义的方法用this换成that
Cannot read properties of undefined (reading ‘row‘)
Cannot read properties of undefined (reading ‘row‘)
下一篇
开通oss服务