14 个折磨人的 JavaScript 面试题

翻到了这篇解释Javascript quiz的文章quiz-legend,反正没事儿,就想搬过来供大家学习、理解、背诵、批判。

问题一

 (function(){
 return typeof arguments;//"object"
})();

arguments是一个Array-like对象,对应的就是传入函数的参数列表。你可以在任何函数中直接使用该变量。
typeof操作符只会返回string类型的结果。参照如下列表可知对应不同数据,typeof返回的值都是什么:

由此我们推断出,typeof arguments是object

问题二

 var f = function g(){ return 23; };
typeof g();//报错

这是一个名字是g的function expression,然后又被赋值给了变量f。
 这里的函数名g和被其赋值的变量f有如下差异:
 函数名g不能变动,而变量f可以被重新赋值
 函数名g只能在函数体内部被使用,试图在函数外部使用g会报错的

问题三

 (function(x){
 delete x;
 return x;//1
})(1);

delete操作符可以从对象中删除属性,正确用法如下:
delete object.property
delete object['property']
delete操作符只能作用在对象的属性上,对变量和函数名无效。也就是说delete x是没有意义的。

你最好也知道,delete是不会直接释放内存的,她只是间接的中断对象引用

问题四
var y = 1, x = y = typeof x; x;//"undefined"
我们试图分解上述代码成下面两步:
var y = 1; //step 1
var x = y = typeof x; //step 2
第一步应该没有异议,我们直接看第二步
 1.赋值表达式从右向左执行
 2.y被重新赋值为typeof x的结果,也就是undefined
 3.x被赋值为右边表达式(y = typeof x)的结果,也就是undefined

问题五

 (function f(f){
 return typeof f();//"number"
})(function(){ return 1; });

直接上注释解释:

 (function f(f){
 //这里的f是传入的参数function(){ return 1; }
 //执行的结果自然是1
 return typeof f(); //所以根据问题一的表格我们知道,typeof 1结果是"number"
})(function(){ return 1; });

问题六

 var foo = {
 bar: function() { return this.baz; },
 baz: 1
};

(function(){
 return typeof arguments[0]();//"undefined"
})(foo.bar);

这里你可能会误以为最终结果是number。向函数中传递参数可以看作是一种赋值,所以arguments[0]得到是是真正的bar函数的值,而不是foo.bar这个引用,那么自然this也就不会指向foo,而是window了。

问题七

 var foo = {
 bar: function(){ return this.baz; },
 baz: 1
}
typeof (f = foo.bar)();//"undefined"

这和上一题是一样的问题,(f = foo.bar)返回的就是bar的值,而不是其引用,那么this也就指的不是foo了。

问题八

var f = (function f(){ return '1'; }, function g(){ return 2; })();
typeof f;//"number"

逗号操作符 对它的每个操作对象求值(从左至右),然后返回最后一个操作对象的值

所以(function f(){ return '1'; }, function g(){ return 2; })的返回值就是函数g,然后执行她,那么结果是2;最后再typeof 2,根据问题一的表格,结果自然是number

问题九

 var x = 1;
if (function f(){}) {
 x += typeof f;
}
x;//"1undefined"

这个问题的关键点,我们在问题二中谈到过,function expression中的函数名f是不能在函数体外部访问的

问题十

 var x = [typeof x, typeof y][1];
typeof typeof x;//"string" 1.因为没有声明过变量y,所以typeof y返回"undefined"

2.将typeof y的结果赋值给x,也就是说x现在是"undefined"
 3.然后typeof x当然是"string"
 4.最后typeof "string"的结果自然还是"string"

问题十一

(function(foo){
 return typeof foo.bar;//"undefined"
})({ foo: { bar: 1 } });

这是个纯粹的视觉诡计,上注释

 (function(foo){

 //这里的foo,是{ foo: { bar: 1 } },并没有bar属性哦。
 //bar属性是在foo.foo下面
 //所以这里结果是"undefined"
 return typeof foo.bar;
})({ foo: { bar: 1 } });

问题十二

 (function f(){
 function f(){ return 1; }
 return f();//2
 function f(){ return 2; }
})();

通过function declaration声明的函数甚至可以在声明之前使用,这种特性我们称之为hoisting。于是上述代码其实是这样被运行环境解释的:

 (function f(){
 function f(){ return 1; }
 function f(){ return 2; }
 return f();
})();

问题十三

function f(){ return f; }
new f() instanceof f;//false

当代码new f()执行时,下面事情将会发生:
 1.一个新对象被创建。它继承自f.prototype
 2.构造函数f被执行。执行的时候,相应的传参会被传入,同时上下文(this)会被指定为这个新实例。new f等同于new f(),只能用在不传递任何参数的情况。
 3.如果构造函数返回了一个“对象”,那么这个对象会取代整个new出来的结果。如果构造函数没有返回对象,那么new出来的结果为步骤1创建的对象,

ps:一般情况下构造函数不返回任何值,不过用户如果想覆盖这个返回值,可以自己选择返回一个普通对象来覆盖。当然,返回数组也会覆盖,因为数组也是对象。

于是,我们这里的new f()返回的仍然是函数f本身,而并非他的实例

问题十四
 with (function(x, undefined){}) length;//2
with语句将某个对象添加的作用域链的顶部,如果在statement中有某個未使用命名空间的变量,跟作用域链中的某個属性同名,則這個变量将指向這個属性值。如果沒有同名的属性,则将拋出ReferenceError异常。

OK,现在我们来看,由于function(x, undefined){}是一个匿名函数表达式,是函数,就会有length属性,指的就是函数的参数个数。所以最终结果就是2了

写在最后

有人觉得这些题坑爹,也有人觉得开阔了眼界,见仁见智吧。但有一件事是真的,无论你是否坚定的实践派,缺了理论基础,也铁定走不远 - 你永远不会见到哪个熟练的技术工人突然成了火箭专家。

看文档、读标准、结合实践,才是同志们的决胜之道。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • 用js实现上传图片前的预览(TX的面试题)

    以前不知道 file 控件也能使用 onchange,导致面试时失去良机. <script> function yulan() { var fileext=document.form1.UpFile.value.substring(document.form1.UpFile.value.lastIndexOf("."),document.form1.UpFile.value.length)         fileext=fileext.toLowerCase() if (

  • 从面试题学习Javascript 面向对象(创建对象)

    题目: 复制代码 代码如下: try{ var me = Man({ fullname: "小红" }); var she = new Man({ fullname: "小红" }); console.group(); console.info("我的名字是:" + me.attr("fullname") + "\n我的性别是:" + me.attr("gender")); consol

  • 你有必要知道的25个JavaScript面试题

    1.使用 typeof bar === "object" 判断 bar 是不是一个对象有神马潜在的弊端?如何避免这种弊端? 使用 typeof 的弊端是显而易见的(这种弊端同使用 instanceof): let obj = {}; let arr = []; console.log(typeof obj === 'object'); //true console.log(typeof arr === 'object'); //true console.log(typeof null

  • BAT及各大互联网公司2014前端笔试面试题--JavaScript篇

    而更多的题目是我一路以来收集的,也有往年的,答案不确保一定正确,如有错误或有更好的解法,还请斧正. 附上第二篇:BAT及各大互联网公司2014前端笔试面试题--Html,Css篇 前面几题是会很基础,越下越有深度. 初级Javascript: 1.JavaScript是一门什么样的语言,它有哪些特点? 没有标准答案. 2.JavaScript的数据类型都有什么? 基本数据类型:String,boolean,Number,Undefined, Null 引用数据类型:Object(Array,Da

  • 网易JS面试题与Javascript词法作用域说明

    调用对象位于作用域链的前端,局部变量(在函数内部用var声明的变量).函数参数及Arguments对象都在函数内的作用域中--这意味着它们隐藏了作用域链更上层的任何同名的属性. 2010年9月14日,我去参加网易网页工程师招聘会,应聘JS工程师职位.有幸参加笔试,然后有幸栽在笔试,呵呵.废话少说,抓出音响极深的一题重新研究研究. 题目大概是:写出如下代码的输出结果并进行分析 复制代码 代码如下: var tt = 'aa'; function test(){ alert(tt); var tt

  • 5个JavaScript经典面试题

    1:Scope作用范围 复制代码 代码如下: (function() {     var a = b = 5;  })();  console.log(b); 什么会被打印在控制台上? 回答 上面的代码会打印 5. 这个问题的诀窍是,这里有两个变量声明,但 a 使用关键字var声明的.代表它是一个函数的局部变量.与此相反,b 变成了全局变量. 这个问题的另一个诀窍是,它没有使用严格模式 ('use strict';).如果启用了严格模式,代码就会引发ReferenceError的错误:B没有定义

  • 一套比较完整的javascript面试题(部分答案)

    一.单选题 1.以下哪条语句会产生运行错误:(a) A.var obj = ();//语法错误 B.var obj = [];//创建数组 C.var obj = {};//创建对象 D.var obj = //; 原因:var obj = new Array ();是对的:JavaScript 中大括号表示创建对象.var obj = { id:1, name:"jacky" };alert(obj.name);上例表示创建一个具有属性 id (值为 1).属性 name(值为 ja

  • JavaScript面试开发常用的知识点总结

    No1.语法和类型  1.声明定义  变量类型:var,定义变量:let,定义块域(scope)本地变量:const,定义只读常量.  变量格式:以字母.下划线"_"或者$符号开头,大小写敏感.  变量赋值:声明但未赋值的变量在使用时值为undefined,未声明的变量直接使用会抛异常.  未赋值变量作计算:结果为NaN.例如:  var x, y = 1; console.log(x + y); //结果为NaN,因为x没有赋值. 2.作用域  变量作用域:在ES6之前没有块声明域,

  • [转]去百度面试的javascript 收获

    这是我记得的问题,基本都没答上来,大家知道的教教小弟,咱不能再不会了 1.在js里类的继承一般是类抄写和原型继承混合使用,在extjs的extend里就是这两种方式混合 但考官问我,这样不是也在子类里加入了对父类函数的引用吗,(会在子类里加上superClass属性)这不是也是污染吗,怎么做更好的,当时被问傻 2.p元素里可不可以加div,不可以为啥,平时从来没想过这个问题 这个我找到了答案 不可以,我试过了 New Document [Ctrl+A 全选 注:如需引入外部Js需刷新才能执行]

  • javascript实现上传图片前的预览(TX的面试题)

    以前不知道 file 控件也能使用 onchange,导致面试时失去良机. <script> function yulan() { var fileext=document.form1.UpFile.value.substring(document.form1.UpFile.value.lastIndexOf("."),document.form1.UpFile.value.length)         fileext=fileext.toLowerCase() if (

随机推荐