中高级JavaScript易错面试题

写出下题的输出

 

1、函数的实参与形参length

 

var length = 10;
function fn() {console.log(this.length);
}
var obj = {length: 5,method: function(fn) {fn();arguments[0]();}
};
console.log(obj.method(fn, 1));  // 0 2

 

 

我们都知道,[1, 2, 3].length可以得到3,"123".length可以得到3,那么函数的length得到什么呢?

function test(a,b,c) {}
test.length // 3
function test(a,b,c,d) {} test.length // 4

可以看到,函数的length似乎返回了参数的个数,那么对于形参和实参有没有区别呢?答案是有。

 

function test() { console.log( arguments.length );}
test(1,2,3); // 输出 3
test(1,2,3,4); // 输出 4

可以看到,在函数中,用arguments.length取到的是函数的实际参数的个数。

另外,我们要知道var length = 10 这样写是不行的,因为length是JavaScript内置的属性,不能用作变量名或函数名。戳这里http://www.runoob.com/js/js-reserved.html查看JavaScript有哪些保留关键字、内置属性等。

所以,当执行fn()时,this.length打印的是fn这个函数的形参的个数,为0;而执行arguments[0]()时,实际上是obj.method()这个方法的arguments调用了fn函数,this.length的this指向的是arguments,他的实际参数个数为2。

 

 

2、函数的解析与预解析过程(变量提升)

 

function fn(a) {console.log(a);     // function a() {alert(1)}var a = 2;function a() {alert(1)}console.log(a);     //2
}
fn(1);

 

这道题还是挺吊炸天的。。我也想了半天。。下面我来讲一下,涉及到函数的解析和预解析过程。

首先遇到function fn这样的函数声明,会进行函数预解析这么个过程,什么是函数预解析?通俗的说就是,从函数体里找变量和函数声明的过程,找到的变量(遇到var就找到了变量)不会去读具体的值,只会赋为undefined;找到的函数声明会赋值为整个函数体,这里有个知识点就是,如果找到的变量和声明同名,那么声明会覆盖变量(我的理解是,毕竟函数体比undefined的强嘛)。

比如此例中,预解析时找到了变量a,并且赋值为undefined,找到了声明function a(){alert(1)},为整个函数体;两者同名,所以声明覆盖了变量a的值,a不再是undefined的,而是函数体。

预解析完成后调用了方法,开始一步一步走方法。首先console.log(a),这时打印出的是函数体;接着var a = 2,a的值从函数体被改成了 2 ;接着是个function a(){}函数声明,注意,声明不能改变变量的值,所以走完这一句,a的值还是2,接着打印出了2。

 

 

有人肯定有这样的疑惑,为什么a=1传进去没起作用呢?这里有一个原则,就是局部变量优先,基于这个原则,我们再来分析一下a的变化过程。预解析中,a=undefined,a=function(){alert(1)},此时参数有值等于1,本应该将a赋值为1,但却没有,原因是此时的a已经等于局部函数声明function(){alert(1)},所以外部传进来的参数1并没有取代a的值;假如本例没有function(){alert(1)}这一句,打印出的将是1,  2。

局部变量优先原则,原理同下:

var a = 5;
function fn(){var a = 10;console.log(a)  // 10,局部变量优先,在局部找到a后,不会再向外查找
}

 

 

3、变量提升、window的变量

if('a' in window) {   var a = 10; 
}  
console.log(a);  // 10

首先,if(){}的花括号并不像function(){}的花括号一样,具有自己的块级作用域,if的花括号还是全局的环境。根据JavaScript的变量提升机制,var a会被js引擎解释到第一行,如下:

var a;
if ('a' in window) {a = 10;
}

接着有个知识点,全局变量是window对象的属性,所以'a'  in  window会返回true,答案就很直白了。

 

这道题我在做的时候踩了个坑,我在代码编辑器里写了如下代码:

window.onload = function(){if('a' in window){var a = 10;}  console.log(a)  // undefined
}

这时候,a这个变量是定义在匿名函数function(){}里的,属于该函数的局部变量,所以a不再是window的对象。大家一定要注意细节。

 

 

4、基本类型无属性

var a = 10;
a.pro = 10;
console.log(a.pro + a);  // NaN
var s = 'hello';
s.pro = 'world';
console.log(s.pro + s)  // undefinedhello

变量a与s都是基本类型,无法给他们添加属性,所以a.pro和s.pro都是undefined。

undefined + 10 得到NaN(not a number)。

undefined + 'hello' 得到undefinedhello,其中undefined被转化为字符串类型。

 

如果实在想给字符串添加属性,我们需要将字符串定义为对象类型的字符串,如下:

var a= new String('objectString')
a.pro = "aaaaaaa"
console.log(a.pro)    // aaaaaaa

 

 

5、async与await的执行

async function sayHello() {console.log('Hello')await sleep(1000)console.log('world!')
}
function sleep(ms) {return new Promise(resolve =>  {console.log("666666");setTimeout(resolve, ms);console.log("888888")})
}
sayHello()  // hello 666666 888888 world!

 

async 表示这是一个async函数,await只能用在这个函数里面。

await 表示在这里等待promise返回结果了,再继续执行。

 

 

首先打出hello,到了await,会等待promise的返回,所以“world”不会立刻打出,接着进入sleep函数,打出666,接着开了一个1秒的定时器,虽然js是单线程的,但setTimeout是异步的,在浏览器中,异步操作都是被加入到一个称为“events loop”队列的地方,浏览器只会在所有同步代码执行完成之后采取循环读取的方式执行这里面的代码,所以resolve被加入任务队列,先打印了888,一秒后执行了resolve,表示promise成功返回,打出了world。

 

 

以上每道题都是本渣自己的想法和理解,如有不正确的地方烦请读者指正,大佬轻喷~

 

转载于:https://www.cnblogs.com/Double-Zhang/p/7777782.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/354620.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

静音设计模式

您最近是否遵循Mute-Design-Pattern™编写了大量代码? 例如 try {complex();logic();here(); } catch (Exception ignore) {// Will never happen heheSystem.exit(-1); }Java 8有一个更简单的方法! 只需将这个非常有用的工具添加到您的Utilities或Hel…

【详细教程】教你如何使用Node + Express + Typescript开发一个应用

Express是nodejs开发中普遍使用的一个框架,下面要谈的是如何结合Typescript去使用。 目标 我们的目标是能够使用Typescript快速开发我们的应用程序,而最终我们的应用程序却是编译为原始的JavaScript代码,以由nodejs运行时来执行。 初始化设置…

结构型模式 适配器模式

结构型模式 适配器模式 适用于: 是将一个类的接口转换成客户希望的另外一个接口。使得原本由于接口不兼容而不能一起工作的那些类可以一起工作。 /*** 结构型模式 适配器模式* Adapter模式也叫适配器模式,是构造型模式之一,通过Adapter模式可…

乐哥学AI_Python(二):Numpy索引,切片,常用函数

Numpy的索引和切片 ndarray对象的内容可以通过索引和切片查看和修改。 索引:ndarray对象中的元素索引基于0开始 切片:对数组里某个片段区域的描述 数组的切片也可以理解为原始数组的局部视图,都是指向内存中的原始数组,所以不同于…

仅使用HTML和CSS实现的标签云效果

标签云的效果在博客和网站上不难见到,它其实就是带有超链接的某些关键字,为了达到强调主题的作用。通常出现概率比较大或者受欢迎的标签文字显示比较大,相反的就显示的小。来源于TagCrowd.com我们就不去深入研究标签云带来的效率上的提升和可…

捍卫Java

因此,我们不时发布了一本电子书,名为“十大Java性能问题” 。 毫无例外,一些人回答说“问题是您正在使用Java”。 显然,Java一直在受到批评,人们已经预测了它的消亡已有一段时间了。 当然,它不像Python&am…

如何在客户端终止一个已经发出的HTTP请求

Javascript 异步编程得益于 Promise 的实现,它们极大地提高了Web开发的性能和体验。不过原生的 Promise 有个最大的缺点就是一旦请求发出去,我们就无法取消它。但是我们找到了另一种方法来实现。DOM 标准中添加了一个称为 AbortController 的新的控制器&…

vuex的使用二

1.先看项目的目录结构 2.在main.js里需要引入store这个文件并挂在实例上 import store from ./store/store ............new Vue({el: #app,router,store,template: <App/>,components: { App } }) 3.store.js里引入action.js和mutation.js文件 // 状态管理器 import Vue…

收到短信验证码自动填充到表单,竟然是这么玩的

苹果系统上的App和网站可以实现来自短信的验证码自动填充表单的功能&#xff0c;通常你是怎么实现这个功能的&#xff1f;有一种实现方式可能你不知道&#xff0c;单纯的HTML标签就能实现&#xff0c;不需要任何的Javascript代码该特性第一次发布是在 WWDC 2018[1]&#xff1a;…

web实现数据交互的几种常见方式

前言在当今社会&#xff0c;作为一名前端程序猿&#xff0c;并不是一昧的去制作静态页面就可以满足滴&#xff1b;你说你会制作网页&#xff0c;好吧&#xff0c;只能说你算是一个前端程序猿。但这是你作为一个程序猿最基本的能力&#xff0c;并不会为你进行加分&#xff1b; 我…

【超详细教程】如何使用TypeScript和GraphQL开发应用

GraphQL是一个专为构建灵活的API而生的强大的查询语言。它允许您为数据定义类型系统&#xff0c;因此在执行查询时&#xff0c;它仅返回所需的数据。与TypeScript一起使用时&#xff0c;GraphQL可以为开发人员提供更好的体验&#xff0c;因为它们都是类型语言。TypeScript是Jav…

jpa jsf_完整Web应用程序Tomcat JSF Primefaces JPA Hibernate –第2部分

jpa jsf托管豆 这篇文章是本教程第1部分的继续。 在“ com.mb”包中&#xff0c;您将需要创建以下类&#xff1a; package com.mb;import org.primefaces.context.RequestContext;import com.util.JSFMessageUtil;public class AbstractMB {private static final String KEEP…

洛谷 P1313 计算系数

题目描述 给定一个多项式(byax)^k&#xff0c;请求出多项式展开后x^n*y^m 项的系数。 输入输出格式 输入格式&#xff1a; 输入文件名为factor.in。 共一行&#xff0c;包含5 个整数&#xff0c;分别为 a &#xff0c;b &#xff0c;k &#xff0c;n &#xff0c;m&#xff0c;…

【利好工具】JavaScript及时运行调试工具

今天要介绍的这个工具叫RunJS[1]&#xff0c;是运行在各大系统上的客户端&#xff0c;目前支持主流的macOS、Windows、Debian和Universal Linux平台。你可以在上面编写你的JavaScript或TypeScript代码&#xff0c;并且获得即时反馈。它的左边是代码区域&#xff0c;右边是输出结…

CIFAR-10 dataset 的下载与使用

基本信息 CIFAR-10 是一个包含60000张图片的数据集。其中每张照片为32*32的彩色照片&#xff0c;每个像素点包括RGB三个数值&#xff0c;数值范围 0 ~ 255。 所有照片分属10个不同的类别&#xff0c;分别是 airplane, automobile, bird, cat, deer, dog, frog, horse, ship, tr…

教你如何创建一款属于自己的VSCode主题

你有没有想过创建一款属于自己的VSCode主题&#xff1f;没有你想像的那么难&#xff0c;但是真正实现起来也不是那么轻松&#xff0c;需要你对将要去改变的主题的一些属性要有所了解和准备。如果你要想让你的主题适用于多种编程语言&#xff0c;并且看起来还很漂亮的话&#xf…

调试

http://www.cnblogs.com/CARPE-DIEM-wu/p/7798119.html 黄金规则 从帮助台得到的观点是不明确的 检查插头 不要想,而要看 回到目录制造失败 回到目录不要想,而要看 回到目录分而治之 回到目录一次只改一个地方 回到目录保持审计跟踪 回到目录检查插头 回到目录获得全新观点 回到…

访问对象的属性,你知道有哪些方法?

相信对象对于每个使用JavaScript的人来说都不陌生。访问对象的属性几乎日常开发每天都在用。下面我们一起探索下有哪些方式可以实现属性的访问。访问对象属性介绍对象是JavaScript的一种常见的数据类型。它允许我们以键值对的形式存储数据。这一系列的键成为对象的属性。常见的…

一些行家技巧和窍门

我正在将使用WebLogic Workshop&#xff08;是的&#xff0c;使用不受支持的IDE可以正确阅读&#xff09;的现有应用程序迁移到Maven。 在旅途中有一些陷阱&#xff0c;我想在这里写下给那些可能会觉得有用并且特别适合自己的人以作为参考。 整个应用程序使用Apache XMLBeans处…

【视频内含福利】原来手机套壳视频是这么做出来的

最近视频圈子里都在发这种视频&#xff0c;究竟是怎么做出来的&#xff1f;研究下来发现原来是用的这款名字叫Record Maker的APP&#xff0c;旨在帮助用户轻松快速给视频加上手机壳的效果。目前App Store上的版本是1.2.1&#xff0c;最近一次更新支持了iPhone11/iPhone12模型&a…