JavaScript 变量不能被 delete 的原因

简介: JavaScript 变量不能被 delete 的原因

前言


请记住:

任何时候,变量只能通过使用 varletconst 关键字才能被声明。


正文



我们都知道, 无论在全局上下文,还是其他任何上下文中,都可以通过省略变量声明关键字的形式(类似 a = 1 )给全局对象添加一个新“属性”。请注意,严格来说这个 a 不能称作“变量”,原因是它不符合 ECMAScript 规范中变量的概念。


举个例子:

var a = 1
b = 2
delete a
delete b
console.log(a) // 1
console.log(b) // ReferenceError: b is not defined

严格来说,示例中 b 不能被称为变量。它只是全局对象下的一个属性。

示例中,变量 a 并没有被删除掉,而 b 却被删除了,为什么呢?问题先放一放


delete 操作符


我们先看看 delete 操作符是怎样工作的,语法如下:

delete object.property
delete object['property']


它的作用是删除对象的某个属性。一般情况下它的返回值都是 true,表示对象某个属性被移除成功。若该属性是一个自身的不可配置的属性,这时非严格模式下返回 false(即属性移除失败),而严格模式下会抛出 TypeError

上面这句话有两个关键词:“自身”、“不可配置”。

  • 自身:是指对象本身的属性,而非原型上的。可通过 Object.prototype.hasOwnProperty() 判断。
  • 不可配置:其实对象属性都有一个属性描述对象,属性描述对象其中包含一个 configurable 属性,若为 false,则表示该属性不可配置。可通过 Object.getOwnPropertyDescriptor() 获取。

var a = 1
b = 2
window.hasOwnProperty('a') // true
window.hasOwnProperty('b') // true
Object.getOwnPropertyDescriptor(window, 'a') // { configurable: false, ... }
Object.getOwnPropertyDescriptor(window, 'b') // { configurable: true, ... }

var obj = {}
obj.__proto__.prop = 'proto'
delete obj.prop // true,尽管 obj 本身没有 prop 属性,仍会返回 true
obj.prop // "proto",delete 不会删除原型上的属性


区别


回到文中开头的问题:变量 a 为什么没有被删除掉?

尽管 var a = 1b = 2 都会往 window 对象下添加对应属性,但是变量相比于简单属性来说,变量有一个特性:DontDelete,这个特性的含义就是不能通过 delete 操作符直接删除变量属性。


eval


另外还要注意,eval 上下文中的变量声明和函数声明没有 DontDelete 特性,因此是可以删除的。

eval('var a = 1; function foo() {}')
delete a // true
delete foo // true


但是 eval 上下文内的函数体内部的变量或函数是不能删除的。

eval('function foo() { var inner = 2; console.log(delete inner); console.log(inner) }')
foo() // false


总结


  • 全局上下文或函数上下文中,变量声明或函数声明总含有 DontDelete 特性。
  • eval 上下文的变量声明或函数声明不含 DontDelete 特性,因此可以被 delete 删除。
  • 不要相信全局对象下的 delete 操作。
  • 编写代码时,不要写出类似 a = 1 省略声明关键字的语句。
目录
相关文章
|
2月前
|
JavaScript 前端开发
JavaScript如何判断变量undefined
JavaScript如何判断变量undefined
|
2月前
|
JavaScript 前端开发
JavaScript基础知识-变量的声明提前
关于JavaScript变量声明提前特性的基础知识介绍。
45 0
JavaScript基础知识-变量的声明提前
|
14天前
|
前端开发 JavaScript
如何在 JavaScript 中访问和修改 CSS 变量?
【10月更文挑战第28天】通过以上方法,可以在JavaScript中灵活地访问和修改CSS变量,从而实现根据用户交互、页面状态等动态地改变页面样式,为网页添加更多的交互性和动态效果。在实际应用中,可以根据具体的需求和场景选择合适的方法来操作CSS变量。
|
14天前
|
前端开发 JavaScript 数据处理
CSS 变量的作用域和 JavaScript 变量的作用域有什么不同?
【10月更文挑战第28天】CSS变量和JavaScript变量虽然都有各自的作用域概念,但由于它们所属的语言和应用场景不同,其作用域的定义、范围、覆盖规则以及与其他语言特性的交互方式等方面都存在明显的差异。理解这些差异有助于更好地在Web开发中分别运用它们来实现预期的页面效果和功能逻辑。
|
14天前
|
前端开发 JavaScript UED
如何使用 JavaScript 动态修改 CSS 变量的值?
【10月更文挑战第28天】使用JavaScript动态修改CSS变量的值可以为页面带来更丰富的交互效果和动态样式变化,根据不同的应用场景和需求,可以选择合适的方法来实现CSS变量的动态修改,从而提高页面的灵活性和用户体验。
|
2月前
|
存储 前端开发 JavaScript
前端基础(二)_JavaScript变量、JavaScript标识符、JavaScript获取元素、JavaScript的鼠标事件
本文介绍了JavaScript变量的声明和使用、标识符的命名规则、如何获取和操作HTML元素,以及JavaScript的鼠标事件处理,通过示例代码展示了这些基础知识点在实际开发中的应用。
43 2
前端基础(二)_JavaScript变量、JavaScript标识符、JavaScript获取元素、JavaScript的鼠标事件
|
26天前
|
JavaScript 前端开发
局部 JavaScript 变量
JavaScript 中,函数内部使用 `var` 声明的变量为局部变量,仅在函数内可见,函数执行完毕后被删除。全局变量则在函数外部声明,整个页面的脚本和函数均可访问,页面关闭后才被删除。未声明的变量赋值会自动成为 `window` 对象的属性,且在非严格模式下可被删除。
|
2月前
|
JavaScript 前端开发
js 变量作用域与解构赋值| 22
js 变量作用域与解构赋值| 22
|
1月前
|
JavaScript 前端开发
什么是JavaScript变量?
什么是JavaScript变量?
34 0
|
2月前
|
存储 JavaScript 前端开发