首页 前端开发 Web前端入门第 55 问:JavaScript 严格模式与非严格模式区别

Web前端入门第 55 问:JavaScript 严格模式与非严格模式区别

JavaScript 默认是非严格模式的,可以通过 "use strict"; 启用严格模式。此声明语句可以放在 JS 文件顶部,也可以放在函数内部。

启用严格模式
1、外部脚本在 JS 文件开头声明,内部脚本在 <script> 标签开头声明,声明后所有 JS 代码启用严格模式:

"use strict";

console.log('Hello World!');
错误写法:

console.log("test");
"use strict"; // 置于代码之后的声明,严格模式不生效!
2、在函数内部声明函数启用严格模式,此时本函数体内部的 JS 代码将启用严格模式:

function strictFunc() {
"use strict"; // 函数级严格模式
let x = 10;
y = 20; // ReferenceError: y is not defined
}

function nonStrictFunc() {
y = 20; // 非严格模式下隐式创建全局变量(不推荐)
}
错误写法:

if (true) {
"use strict"; // 无效!严格模式无法在块级作用域启用
x = 10; // 非严格模式下隐式创建全局变量
}
3、模块化脚本(ES6 Modules)无需显式声明,默认启用严格模式:

// module.js(无需写 "use strict")
export default function() {
x = 10; // ReferenceError: x is not defined
}
4、类(ES6 Class)声明或类方法内部无需显式声明默认启用严格模式:

class MyClass {
constructor() {
x = 10; // ReferenceError: x is not defined
}
}
为何 JS 代码会有两种不同的解析结果?

这就不得不提到历史原因了,JS 之父创造 JavaScript 时,仅用了 10 天时间,这久导致了 JS 在后来使用中发现了一些问题,又由于浏览器的版本迭代必须要兼顾一些旧的代码(不可能浏览器来一个版本更新,直接把所有网站一棒打死),所以就有了 严格模式 的出现,这个模式的用途就是告诉浏览器,我这个网站的代码你按照 严格模式 来解析,无需考虑历史兼容性,可能存在的隐式错误可以先告诉我。

严格模式 vs 非严格模式
主要区别如下:

变量声明
非严格模式:未声明的变量赋值会隐式创建全局变量。

严格模式:未声明的变量赋值会抛出 ReferenceError。

<script>
function test1 () {
y = 10; // 声明为全局变量
}
test1();
console.log(y); // 10
</script>
<script>
function test2 () {
"use strict";
x = 10; // ReferenceError: x is not defined
}
test2();
console.log(x); // test2 报错,此行代码不执行
</script>
静默错误转显式错误
删除不可删除的属性:delete Object.prototype; 在严格模式下报错。

重复参数名:function(a, a) {} 在严格模式下报语法错误。

只读属性赋值:NaN = 1; 在严格模式下报错。

<script>
function test1 (a, a) {
delete Object.prototype;
NaN = 1;
}
test1();
</script>
<script>
"use strict";
function test2 (a, a1) { // 报错 SyntaxError
delete Object.prototype; // 报错 TypeError
NaN = 1; // 报错 TypeError
}
test2();
</script>
this 指向
非严格模式:全局函数中 this 指向全局对象(如 window)。

严格模式:全局函数中 this 为 undefined。

<script>
function test1 () {
console.log(this); // 浏览器中指向 Window 对象,nodejs 中指向 global 对象
}
test1();
</script>
<script>
"use strict";
function test2 () {
console.log(this); // undefined
}
test2();
</script>
eval 和 arguments 限制
eval 变量泄漏:严格模式下 eval 中的变量不会污染外部作用域。

禁用 arguments.callee:防止递归调用导致性能问题。

<script>
function test1 () {
eval('var a = 10');
console.log(arguments.callee); // 指向函数本身
console.log(a); // 10
}
test1();
</script>
<script>
"use strict";
function test2 () {
eval('var a = 10');
console.log(arguments.callee); // 报错 TypeError: 'caller', 'callee', and 'arguments' properties may not be accessed on strict mode functions or the arguments objects for calls to them
console.log(a); // ReferenceError: a is not defined
}
test2();
</script>
其他限制
八进制表示:禁止 0123,需用 0o123。

with 语句:严格模式下禁用,避免作用域混乱。

保留字限制:如 interface、private 等不能作为变量名。

<script>
function test1 () {
console.log(0123); // 83
with (Math) {
console.log(random()); // 输出随机数
}
var interface = 'str';
console.log(interface); // str
}
test1();
</script>
<script>
"use strict";
function test2 () {
console.log(0123); // SyntaxError: Octal literals are not allowed in strict mode.
with (Math) { // SyntaxError: Strict mode code may not include a with statement
console.log(random());
}
var interface = 'str'; // SyntaxError: Unexpected strict mode reserved word
console.log(interface);
}
test2();
</script>
写在最后
建议始终启用严格模式,强制更安全的编码实践,避免隐式错误,提升代码质量。

站心网

JavaScript 默认是非严格模式的,可以通过 "use strict"; 启用严格模式。此声明语句可以放在 JS 文件顶部,..

为您推荐

Web前端入门第 53 问:JavaScript 的各种调试方法

任何一门编程语言,在学习之前都应该先弄清楚它的调试方法,毕竟没有不挖坑的人类!程序一旦出现问题,第一时间就是找到问题出在哪儿,其次才是拿出解决办法。如果都找不到问题原因,那又何从谈起解决办法呢?如何排..

Web前端入门第 54 问:JavaScript 3 种书写位置及 script 标签的正确存放位置

JS 的代码并没有强制规定放在 HTML 中的某个位置,如果您有使用过开发者工具查看过网页源码,那么您会看到很多 JS 代码都以 .js 文件的形式存放,并且放在了 HTML 文件最后,也就是 </body> 结束标签之前。但如果仔..

Web前端入门第 52 问:JavaScript 的应用领域

截至目前,您应该对前端的 HTML + CSS 应该有了很清楚的认知,至少实现一个静态网页已经完全不在话下了。当然,CSS 功能绝不止这些,一些不太常用的 CSS 相关知识,后续将通过案例进行分享。那么咱们接下来看看 Java..

.NET C# 过滤从富文本编辑器html里的Javascript脚本

富文本编辑器在允许用户输入丰富内容的同时,也带来了跨站脚本攻击(XSS)的风险。过滤提交的 HTML 中的 <script> 脚本是防止跨站脚本攻击(XSS)的关键步骤。在 .NET C# 服务端过滤 <script> 脚本主要有以下几种方..

轻松学习 JavaScript函数中的默认参数

JavaScript函数可以有默认参数值。通过默认函数参数,你可以初始化带有默认值的正式参数。如果不初始化具有某些值的参数,则该参数的默认值为undefined。请看下列代码:function foo(num1){console.log(num1);}foo()..

JavaScript API 设计原则详解

前言本篇博文来自一次公司内部的前端分享,从多个方面讨论了在设计接口时遵循的原则,总共包含了七个大块。系卤煮自己总结的一些经验和教训。本篇博文同时也参考了其他一些文章,相关地址会在后面贴出来。很难做到详..

JavaScript 中精度问题以及解决方案

JavaScript 中的数字按照 IEEE 754 的标准,使用 64 位双精度浮点型来表示。其中符号位 S,指数位 E,尾数位M分别占了 1,11,52 位,并且在ES5 规范中指出了指数位E的取值范围是[-1074, 971]。精度问题汇总想用有限..

JavaScript 六种继承方式

继承是面向对象编程中又一非常重要的概念,JavaScript支持实现继承,不支持接口继承,实现继承主要依靠原型链来实现的。原型链首先得要明白什么是原型链,在一篇文章看懂proto和prototype的关系及区别中讲得非常详细..

JavaScript 事件委托详解

基本概念事件委托,通俗地来讲,就是把一个元素响应事件(click、focus……)的函数委托到另一个元素;一般来讲,会把一个或者一组元素的事件委托到它的父层或者更外层元素上,真正绑定事件的是外层元素,当事件响应..

JavaScript 中的遍历详解

编程这么多年,要是每次写遍历代码时都用 for 循环,真心感觉对不起 JavaScript 语言~对象遍历为了便于对象遍历的测试,我在下面定义了一个测试对象obj。测试对象// 为 Object 设置三个自定义属性(可枚举)Object.p..

值得探索的 8 个机器学习 JavaScript 框架

JavaScript开发人员倾向于寻找可用于机器学习模型训练的JavaScript框架。下面是一些机器学习算法,基于这些算法可以使用本文中列出的不同JavaScript框架来模型训练:简单的线性回归多变量线性回归逻辑回归朴素贝叶斯..

JavaScript 保留两位小数

以下我们将为大家介绍 JavaScript 保留两位小数的实现方法:四舍五入以下处理结果会四舍五入:var num =2.446242342;num = num.toFixed(2); // 输出结果为 2.45不四舍五入以下处理结果不会四舍五入:第一种,先把小数边..

JavaScript 页面跳转、页面重定向

JavaScript 实现页面跳转重定向可以使用以下两种方法:window.location.replace("url")类似 HTTP 重定向将地址替换成新 url,该方法通过指定 URL 替换当前缓存在历史里(客户端)的项目,因此当使用 replace 方法之..

在 Javascript 中 声明时用 var 与不用 var 的区别

avascript 声明变量的时候,虽然用 var 关键字声明和不用关键字声明,很多时候运行并没有问题,但是这两种方式还是有区别的。可以正常运行的代码并不代表是合适的代码。var num = 1;是在当前域中声明变量。如果在方..

javascript中call()、apply()、bind()的用法终于理解

其实是一个很简单的东西,认真看十分钟就从一脸懵B 到完全 理解!先看明白下面:例 1obj.objAge; // 17obj.myFun() // 小张年龄 undefined例 2shows() // 盲僧 比较一下这两者 this 的差别,第一个打印里面的 this ..

5 个顶级的 JavaScript Ajax 组件和库

在这篇文章中,我们将介绍一些用于AJAX调用的最好的JS库,包括jQuery,Axios和Fetch。欢迎查看代码示例!AJAX是用来对服务器进行异步HTTP调用的一系列web开发技术客户端框架。 AJAX即Asynchronous JavaScript and XM..

float 与 double 类型区别

float 单精度浮点数在机内占 4 个字节,用 32 位二进制描述。double 双精度浮点数在机内占 8 个字节,用 64 位二进制描述。浮点数在机内用指数型式表示,分解为:数符,尾数,指数符,指数四部分。数符占 1 位二进制..

5 分钟掌握 JavaScript 实用窍门

简评:一开始 JavaScript 只是为网页增添一些实时动画效果,现在 JS 已经能做到前后端通吃了,而且还是年度流行语言。本文分享几则 JS 小窍门,可以让你事半功倍 ~1. 删除数组尾部元素一个简单方法就是改变数组的len..

CPU、GPU 和 TPU 之间有什么区别?

什么是 CPU、GPU 和 TPU?它们都是用于计算任务的处理器芯片。可以把你的大脑想象成一台计算机,能够完成诸如阅读书籍或解决数学问题的任务。每一项活动都类似于一个计算任务。例如,当你用手机拍照、发送短信或打开..

javascript 6种连接数组的方法和对比

在 JavaScript 中,有多种方法可以用来连接数组,以下是常见的几种:1. concat() 方法用于连接两个或多个数组,返回一个新的数组,不修改原数组。const arr1 = [1, 2, 3];const arr2 = [4, 5, 6];const result = arr..

发表回复

返回顶部

微信分享

微信分享二维码

扫描二维码分享到微信或朋友圈

链接已复制