博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
一道 JavaScript 面试题
阅读量:4341 次
发布时间:2019-06-07

本文共 1657 字,大约阅读时间需要 5 分钟。

有一道 JavaScript 面试题。

f = function () { return true; };g = function () { return false; };(function() {    if (g() && [] == ![]) {        f = function () { return false; };        function  g() { return true; }    }})();console.info(f());

首先看前两行

var f = function () { return true; };var g = function () { return false; };

这两行定义了两个变量,而不是函数定义,变量是可以重新赋值的。

第四行就比较有趣了

(function () {    if (g() && [] == ![]) {        f = function () { return false; };        function g() { return true; }    }})();

在 if 条件中调用了函数 g(),但是,这个 g 是第二行定义的函数 g 吗?答案是否定的,因为在匿名函数中又定义了函数 g,局部的定义覆盖了外部的定义,所以,这里调用的是内部函数 g,返回为 true。第一个条件通过,进入第二个条件。

其实,最有趣的是第二个条件 [] == ![] 。

先看 ![] ,在 JavaScript 中,当用于布尔运算的时候,比如在这里,对象的非空引用被视为 true,空引用 null 则被视为 false。由于这里不是一个 null, 而是一个没有元素的数组,所以 [] 被视为 true, 而 ![] 的结果就是 false 了,如果你在 Visual Studio 的编辑器中输入,就会直接看到提示,表达式永远为假。

当一个布尔值参与到条件运算的时候,true 会被看作 1, 而 false 会被看作 0,见这里的说明:

现在条件变成了 [] == 0 的问题了,当一个对象参与条件比较的时候,它会被求值,求值的结果是数组成为一个字符串,[] 的结果就是 '' ,而 '' 会被当作 0 ,所以,条件成立。

条件内的代码如下:

f = function () { return false; };function g() { return true; }

注意 f 的前面没有 var ,所以,不是局部变量,因此,这里会通过闭包访问到外部的变量 f, 重新赋值,现在执行 f 函数返回值已经成为 false 了。

至于 g 则不会有这个问题,这里是一个函数内定义的 g,不会影响到外部的 g 函数。

最后一行的结果自然也就出来了,就是 false。

---------------------------------------------------------------------------------------------------

今天又同学说到执行的结果不同,我原来是在 Chrome 测试的,现在重新检查一下发现是有问题的。

Chrome 在执行到签到函数的时候,是先发现 g 函数重新定义了,覆盖了外部的定义,满足条件执行,就是上面的说明。

在 FireFox 中,直到执行到 if 语句之前,是不会看内部的 g 函数定义的,所以条件根本就不会满足,在执行条件判断之后,才会处理内部的函数定义,但是由于条件没有满足,所以 f 不会被重新定义。返回结果就成为 true.

在 IE 中,最为特殊,在进入内部函数之后之后,g 既不是外部的定义,也不是内部的定义,而是神奇地成为 undefined 。直接会抛出异常。如果内部没有定义函数 g就可以访问到外部的 g 函数。

转载于:https://www.cnblogs.com/haogj/p/3926715.html

你可能感兴趣的文章
小D课堂 - 零基础入门SpringBoot2.X到实战_第10节 SpringBoot整合定时任务和异步任务处理_41、SpringBoot定时任务schedule讲解...
查看>>
小D课堂 - 零基础入门SpringBoot2.X到实战_第10节 SpringBoot整合定时任务和异步任务处理_43、SpringBoot2.x异步任务实战(核心知识)...
查看>>
小D课堂 - 新版本微服务springcloud+Docker教程_1_01课程简介
查看>>
小D课堂 - 零基础入门SpringBoot2.X到实战_第11节 Logback日志框架介绍和SpringBoot整合实战_45、SpringBoot2.x日志讲解和Logback配置实战...
查看>>
小D课堂 - 新版本微服务springcloud+Docker教程_3-05 服务注册和发现Eureka Server搭建实战...
查看>>
小D课堂 - 新版本微服务springcloud+Docker教程_4-05 微服务调用方式之feign 实战 订单调用商品服务...
查看>>
UI基础--烟花动画
查看>>
Android dex分包方案
查看>>
ThreadLocal为什么要用WeakReference
查看>>
删除本地文件
查看>>
FOC实现概述
查看>>
base64编码的图片字节流存入html页面中的显示
查看>>
这个大学时代的博客不在维护了,请移步到我的新博客
查看>>
GUI学习之二十一——QSlider、QScroll、QDial学习总结
查看>>
gethostbyname与sockaddr_in的完美组合
查看>>
kibana的query string syntax 笔记
查看>>
旋转变换(一)旋转矩阵
查看>>
thinkphp3.2.3 bug集锦
查看>>
[BZOJ 4010] 菜肴制作
查看>>
C# 创建 读取 更新 XML文件
查看>>