形参与变量声明和函数声明同名(自测原型链)

题目来源:https://www.cnblogs.com/simonbaker/p/4237970.html
九道题目还是很有纪念价值的:
第一道:

a()
var a = c = function() {
    console.log(2)
}
a()

function a() {
    console.log(1)
}
a();
(function(b) {
    b(), c()
    var b = c = function a() {
        console.log(3)
    }
    b()
})(a)
c()

答案:1 2 2 2 2 3 3
解析:这个可以放在最后解析

var A = function() {}
A.prototype.n = 1;

var b = new A()

A.prototype = {
    n: 2,
    m: 3
}

var c = new A()
console.log(b.n, b.m, c.n, c.m)

答案:1 undefined 2 3
解析:在b被实例化的时候只是有A的变量表达式,给了A的原型对象一个n的属性,此时m属性为定义,因此b.m=undefined;类似原理,在c被实例化的时候,A的原型对象被重新赋值,因此c.n=2,c.m=3

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

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

答案:2
解析:同名函数声明,后面的函数声明替换前面的函数声明,因此返回的f()是第二个声明的函数

if (!a) {
    var a = 1;
}
console.log(a)

答案: 1
解析:这个是不是很简单,因为js没有块级作用域呀。
拓展一下:

if (!(a in window)) {
    var a = 1;
}
console.log(a)

变量提升,因此首先a在window中满足条件,因此赋值语句无法执行,因此最后打印出来是undefined。
如果此时var变为let,那么无法打印a,因为let是可以形成块级作用域呀,无法提升变量!!!


image.png

5

function a() {}
var a;
console.log(typeof a)

答案:function
解析:只要记住一条就可以了,函数声明 替换 变量声明(记住不要把变量声明理解为变量赋值哦)
6

(function(b) {
    console.log(b)
    var b = c = 2
    console.log(b)
})(1)

答案:1 2
解析:

(function (b) {
    var b = 1//所谓的传入参数的过程就是声明加赋值
    console.log(b)
    var b = c = 2
    console.log(b)
})(1)

7

(function(b) {
    console.log(b)
    var b = c = 2
    console.log(b)

    function b() {}
    console.log(b)
})(1)

答案:f b(){} 2 2
解析:换一种写法就很清楚了

(function(b) {
    var b=1//参数定义搁在最前面
    function b() {}//函数声明提升,不好意思,把上面的覆盖了
    console.log(b)//打印的就是函数喽
    var b = c = 2
    console.log(b)//2
    console.log(b)//2
})(1)
var a = 10;

function fn() {
    console.log(1)
    var a = 100;
    console.log(a)
}
fn();

答案:1 100
解析:这个要是不知道,那就去看书吧
9

var a = 1

function c(a, b) {
    console.log(a)
    a = 2
    console.log(a)
}
c()

答案:undefined 2
解析:

var a = 1

function c(a, b) {
    console.log(a)
    a = 2
    console.log(a)
}
c()
console.log(a)

打印一下a就会发现是1呀,所以参数如果和外部的变量同名,不要去纠结,把参数当做另一个字母表示就可以了

在讲解题目的时候,先贴两份代码,

当传入参数为函数声明的时候:

function f(a) {
    console.log(a);  // ƒ a() {  console.log('outer'); }
    var a = 2;
    console.log(a);  // 2
}
f(a);
function a() {
    console.log('outer');
}

换个清楚的:

function f(a) {
    function a() {
      console.log('outer');
    }
    console.log(a);  // ƒ a() {  console.log('outer'); }
    var a = 2;
    console.log(a);  // 2
}
f(a);
function a() {
    console.log('outer');
}

再来一个

function f(a) {
    console.log(a);  // ƒ a() { }
    var a = 2;
    function a() {}
    console.log(a);  // 2
}
f(a);
function a() {
    console.log('outer');
}

擦擦眼睛:

function f(a) {
function a() {
    console.log('outer');
}
    function a() {}
    console.log(a);  // ƒ a() { }
    var a = 2;
    console.log(a);  // 2
}
f(a);
function a() {
    console.log('outer');
}

第二类情况:传入函数赋值的变量


function f(a) {
    console.log(a);  // function a() {}
    var a = 2;
    function a() {}
    console.log(a);  // 2
}
f(a);
var a = function b() {
    console.log('outer');
}

擦擦眼睛:

function f(a) {
    var a = function b() {
      console.log('outer');
    }
    function a() {}
    console.log(a);  // function a() {}
    var a = 2;
    console.log(a);  // 2
}
f(a);
var a = function b() {
    console.log('outer');
}

ok!!再回头看第一题是不是很好理解了
解析:

function a() {
    console.log(1)
} //函数声明提升
a() //第一个a()打印出来就是1
var a = c = function () {
    console.log(2)
} //给a,c重新赋值
a() //第二个a执行因此为2

a(); //第三个a执行因此为2
(function (b) { //b其实就是a啊
    b(), c() //a没有改变,当然打印2呀,c又不是参数,c根据作用域链,发现了父级的作用域链,打印的当然也是2
    var b = c = function a() {
        console.log(3)
    } //记得赋值是从右往左赋值,而且c被重新赋值了,而且c还在window中
    b()
})(a)
c()
a()//2

完结!!!
参考:https://www.cnblogs.com/simonbaker/p/4237970.html
https://www.cnblogs.com/aredleave/p/7596233.html
https://blog.csdn.net/weixin_34273046/article/details/87015464

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,657评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,662评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,143评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,732评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,837评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,036评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,126评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,868评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,315评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,641评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,773评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,859评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,584评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,676评论 2 351

推荐阅读更多精彩内容