前端工程师有时候面试时会遇到一类面试官,他们问的问题对于语言自己很是较真儿,每每不是候选人可能期待的面向实际的问题(有些候选人强调能干活就行,至于知不知道其中原因是无关痛痒的)。这类题目,虽然没有逻辑,但某种程度说,确实考察了候选人对于javascript
这门语言的理解。javascript
忽然想到这个话题是无聊在翻本身的Github,看看之前都写过什么丑货。而后翻到了这篇解释Javascript quiz的文章quiz-legend,反正没事儿,就想搬过来供你们学习、理解、背诵、批判。前端
(function(){ return typeof arguments;//"object" })();
arguments
是一个Array-like对象,对应的就是传入函数的参数列表。你能够在任何函数中直接使用该变量。java
typeof
操做符只会返回string
类型的结果。参照以下列表可知对应不一样数据,typeof
返回的值都是什么:git
类型 | 结果 |
---|---|
undefined |
'undefined' |
null |
'object' |
Boolean |
'boolean' |
Number |
'number' |
String |
'string' |
Symbol (new in ECMAScript 2015) | 'symbol' |
Host object (provided by the JS environment) | Implementation-dependent |
Function object (implements [[Call]] in ECMA-262 terms) | 'function' |
Any other object | 'object' |
由此咱们推断出,
typeof arguments
是object
github
var f = function g(){ return 23; }; typeof g();//报错
这是一个名字是g
的function expression,而后又被赋值给了变量f
。面试
这里的函数名g
和被其赋值的变量f
有以下差别:express
函数名g
不能变更,而变量f
能够被从新赋值数组
函数名g
只能在函数体内部被使用,试图在函数外部使用g
会报错的前端工程师
(function(x){ delete x; return x;//1 })(1);
delete
操做符能够从对象中删除属性,正确用法以下:ide
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
第一步应该没有异议,咱们直接看第二步
赋值表达式从右向左执行
y
被从新赋值为typeof x
的结果,也就是undefined
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"
由于没有声明过变量y
,因此typeof y
返回"undefined"
将typeof y
的结果赋值给x
,也就是说x
如今是"undefined"
而后typeof x
固然是"string"
最后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()
执行时,下面事情将会发生:
一个新对象被建立。它继承自f.prototype
构造函数f
被执行。执行的时候,相应的传参会被传入,同时上下文(this
)会被指定为这个新实例。new f
等同于new f()
,只能用在不传递任何参数的状况。
若是构造函数返回了一个“对象”,那么这个对象会取代整个new
出来的结果。若是构造函数没有返回对象,那么new
出来的结果为步骤1建立的对象,
ps:通常状况下构造函数不返回任何值,不过用户若是想覆盖这个返回值,能够本身选择返回一个普通对象来覆盖。固然,返回数组也会覆盖,由于数组也是对象。
因而,咱们这里的new f()
返回的仍然是函数f
自己,而并不是他的实例
with (function(x, undefined){}) length;//2
with
语句将某个对象添加的做用域链的顶部,若是在statement
中有某個未使用命名空间的变量,跟做用域链中的某個属性同名,則這個变量将指向這個属性值。若是沒有同名的属性,则将拋出ReferenceError
异常。
OK,如今咱们来看,因为function(x, undefined){}
是一个匿名函数表达式,是函数,就会有length
属性,指的就是函数的参数个数。因此最终结果就是2
了
有人以为这些题坑爹,也有人以为开阔了眼界,见仁见智吧。但有一件事是真的,不管你是否坚决的实践派,缺了理论基础,也铁定走不远 - 你永远不会见到哪一个熟练的技术工人忽然成了火箭专家。
看文档、读标准、结合实践,才是同志们的决胜之道