中高级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 //
可以看到,函数的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); //
}
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。
以上每道题都是本渣自己的想法和理解,如有不正确的地方烦请读者指正,大佬轻喷~
中高级JavaScript易错面试题的更多相关文章
- JavaScript易错知识点
JavaScript易错知识点整理1.变量作用域上方的函数作用域中声明并赋值了a,且在console之上,所以遵循就近原则输出a等于2. 上方的函数作用域中虽然声明并赋值了a,但位于console之下 ...
- JavaScript易错知识点整理
前言 本文是我学习JavaScript过程中收集与整理的一些易错知识点,将分别从变量作用域,类型比较,this指向,函数参数,闭包问题及对象拷贝与赋值这6个方面进行由浅入深的介绍和讲解,其中也涉及了一 ...
- JavaScript 易错知识点整理
本文是我学习JavaScript过程中收集与整理的一些易错知识点,将分别从变量作用域,类型比较,this指向,函数参数,闭包问题及对象拷贝与赋值这6个方面进行由浅入深的介绍和讲解,其中也涉及了一些ES ...
- JavaScript易错点转载
前言 本文是我学习JavaScript过程中收集与整理的一些易错知识点,将分别从变量作用域,类型比较,this指向,函数参数,闭包问题及对象拷贝与赋值这6个方面进行由浅入深的介绍和讲解,其中也涉及了一 ...
- 《java入门第一季》之面向对象(一个易错面试题)
这个面试题有点难度,有一些饶.不明白可以在下面讨论.还是值得搞懂的. / * 看程序写结果: A:成员变量的问题 int x = 10; //成员变量x是基本类型 Student s = new St ...
- JavaScript易错知识点整理[转]
前言 本文是我学习JavaScript过程中收集与整理的一些易错知识点,将分别从变量作用域,类型比较,this指向,函数参数,闭包问题及对象拷贝与赋值这6个方面进行由浅入深的介绍和讲解,其中也涉及了一 ...
- JavaScript易错点
JavaScript知识点1.变量作用域 var a = 1;function test() { var a = 2; console.log(a); // 2} test(); ...
- 10道C++输出易错笔试题收集
下面这些题目都是我之前准备笔试面试过程中积累的,大部分都是知名公司的笔试题,C++基础薄弱的很容易栽进去.我从中选了10道简单的题,C++初学者可以进来挑战下,C++大牛也可以作为娱乐玩下(比如下面的 ...
- JavaScript易错点 -- 数组比较
记得当初初学JavaScript时尝试用“==”或“===”比较两个数组是否相等, var a = [1,2,3] var b = [1,2,3] if(a == b){ //false //do s ...
随机推荐
- String类的常见面试题(3)
1.判断定义为String类型的s1和s2是否相等 String s1 = "abc"; //这个"abc"对象首先会进常量池 String s2 = &quo ...
- shell脚本获取文件中key/value的小方法
方法有N种,awk.sad.grep.cut... 以上几种方式不写了,就写两个不太常用到的. 废话少说,直接上代码: cat a.txt aa.gif=aaaa.gif bb.gif=bbbb.gi ...
- unable to dequeue a cell with identifier MealTableViewCell
1 问题描述 Terminating app due to uncaught exception 'NSInternalInconsistencyException', reason: 'unable ...
- 一个基于Asp.net MVC的博客类网站开源了!
背景说明: 大学时毕业设计作品,一直闲置在硬盘了,倒想着不如开源出来,也许会对一些人有帮助呢,而且个人觉得这个网站做得还是不错了,毕竟是花了不少心思,希望对你有所帮助. github地址:https: ...
- Application->ProcessMessages();
Application.ProcessMessages的用法意义 在循环中加Application.ProcessMessages是可以防止其他控件没响应,举个例子容易明白:假如你的窗体上有两个按 ...
- windows下实现linux的远程访问以及linux上文件的上传和下载
在网络性能.安全性.可管理性上,Linux有着其他系统无法比拟的强大优势,而服务器对这些方面要求特别高,因此Linux常常被用来做服务器使用.而当我们需要维护linux服务器的时候,就需要远程访问li ...
- 80端口被system 占用
1 运行'netstat -ano'发现80端口被pid=4的进程占用 2 打开任务管理器,发现pid=4的进程,其实是system进程,其对应的进程描述是NT kernel & system ...
- shim 和 polyfill
在前端,有两个词经常被提及:shim 和 polyfill.最近在翻译文章时又遇到了 polyfill 这个词,准备把这两个概念理清楚. 关于 JavaScript 的兼容性问题,通常有不同的解决方案 ...
- Codecraft-17 and Codeforces Round #391 (Div. 1 + Div. 2, combined)D. Felicity's Big Secret Revealed
题目连接:http://codeforces.com/contest/757/problem/D D. Felicity's Big Secret Revealed time limit per te ...
- 前端基础之初识 HTML
HTML HTML(Hypertext Markup Language)即超文本标记语言,是WWW的描述语言.设计HTML语言的目的是为了能把存放在一台电脑中的文本或图形与另一台电脑中的文本或图形方便 ...