JS额外知识补充

37 阅读4分钟

with语句

with语句 扩展一个语句的作用域链。

"use strict";

var message = "Hello World"
// console.log(message)

// with语句: 可以形成自己的作用域
var obj = {name: "why", age: 18, message: "obj message"}

function foo() {
  function bar() {
    with(obj) {
      console.log(message)  // obj message
    }
  }
  bar()
}

foo()

var info = {name: "kobe"}
with(info) {
  console.log(name) // kobe
}

with语句在老版本的vue中有使用过,但是新版本的vue中已经不使用了,而且在严格模式下使用with语句会报错,所以了解一下即可。 不建议使用with语句,因为它可能是混淆错误和兼容性问题的根源。

eval函数

eval是一个特殊的函数,它可以将传入的字符串当做JavaScript代码来运行。

var jsString = 'var message = "Hello World"; console.log(message);'

eval(jsString) // 不会做优化

// 如果是如下普通的JS代码,JS引擎会做优化
var message = "Hello World"
console.log(message)

不建议在开发中使用eval:

  • eval代码的可读性非常的差(代码的可读性是高质量代码的重要原则);
  • eval是一个字符串,那么有可能在执行的过程中被刻意篡改,那么可能会造成被攻击的风险;
  • eval的执行必须经过JS解释器,不能被JS引擎优化;

什么是严格模式

在ECMAScript5标准中,JavaScript提出了严格模式的概念(Strict Mode):

严格模式很好理解,是一种具有限制性的JavaScript模式,从而使代码隐式的脱离了 ”懒散(sloppy)模式“;支持严格模式的浏览器在检测到代码中有严格模式时,会以更加严格的方式对代码进行检测和执行;

严格模式对正常的JavaScript语义进行了一些限制:

  • 严格模式通过 抛出错误 来消除一些原有的 静默(silent)错误;
  • 严格模式让JS引擎在执行代码时可以进行更多的优化(不需要对一些特殊的语法进行处理);
  • 严格模式禁用了在ECMAScript未来版本中可能会定义的一些语法;

开启严格模式

那么如何开启严格模式呢?严格模式支持粒度话的迁移:

  • 可以支持在js文件中开启严格模式;
  • 也支持对某一个函数开启严格模式;

严格模式通过在文件或者函数开头使用 use strict 来开启。

严格模式有哪些限制

这里我们来说几个严格模式下的严格语法限制,JavaScript被设计为新手开发者更容易上手,所以有时候本来错误语法,被认为也是可以正常被解析的;但是这种方式可能给带来留下来安全隐患;在严格模式下,这种失误就会被当做错误,以便可以快速的发现和修正;

  1. 严格模式禁止意外的创建全局变量
  2. 严格模式会使引起静默失败(silently fail,注:不报错也没有任何效果)的赋值操作抛出异常
  3. 严格模式下试图删除不可删除的属性会报错
  4. 严格模式不允许函数参数有相同的名称
  5. 严格模式不允许使用原先的八进制语法
  6. 严格模式下,不允许使用with
  7. 严格模式下,eval函数不会向上引用变量了
  8. 严格模式下,自执行函数(默认绑定)会指向undefined, 而不是window了
"use strict"

// 1. 禁止意外创建全局变量
message = "Hello World"
console.log(message)

function foo() {
  age = 20
}

foo()
console.log(age)

// 2.不允许函数有相同的参数名称
function foo(x, y, x) {
  console.log(x, y, x)
}

foo(10, 20, 30)

// 3.静默错误
true.name = "abc"
NaN = 123
var obj = {}
Object.defineProperty(obj, "name", {
  configurable: false,
  writable: false,
  value: "why"
})
console.log(obj.name)
// obj.name = "kobe"

delete obj.name // 删除不可删除的属性,会报错

// 4.不允许使用原先的八进制格式 0123
var num = 0o123 // 八进制
var num2 = 0x123 // 十六进制
var num3 = 0b100 // 二进制
console.log(num, num2, num3)

// 5.with语句不允许使用

// 6.eval函数不会向上引用变量了
var jsString = 'var message = "Hello World"; console.log(message);'
eval(jsString)

// 开启严格模式之后这行代码会报错, 因为不会再给上层作用域添加message属性了,所以会报错
console.log(message)

// 7. 在严格模式下, 自执行函数(默认绑定)会指向undefined, 而不是window了
// 这不会对我们以前的代码产生影响,因为之前编写的代码中, 自执行函数我们是没有使用过this去获取window
// 我们都是直接拿到window来使用的
function foo() {
  console.log(this)
}

var obj = {
  name: "why",
  foo: foo
}

foo() // undefined

obj.foo()  // obj
var bar = obj.foo
bar() // undefined