鍍金池/ 問答/HTML/ 為什么將函數(shù)c賦值給變量b,在函數(shù)體里面,給c賦值,為什么會失敗

為什么將函數(shù)c賦值給變量b,在函數(shù)體里面,給c賦值,為什么會失敗

為什么將函數(shù)c賦值給變量b,在函數(shù)體里面,給c賦值,為什么會失?。恳簿褪沁@代碼執(zhí)行時為什么c打印出來的不是3

var  b = function c () {
    a=1, b=2, c=3;
    console.log(a);
    console.log(b);
    console.log(c);
}
b();
回答
編輯回答
練命

可以看下這篇文章,大概意思是命名函數(shù)表達(dá)式NFE的名字是存在一個輔助特殊對象(auxiliary special object)中的,而且是只讀的,所以在非嚴(yán)格模式下修改會靜默失敗,如果這段函數(shù)內(nèi)部在嚴(yán)格模式下運(yùn)行會報錯Uncaught TypeError: Assignment to constant variable.,其他答案中說的c被定義成全局是不對的,可以試一下把console.log(c)改成console.log(window.c)也是undefined。

2017年6月6日 02:08
編輯回答
離觴

感謝@Showonne指出的錯誤,我修改下答案:
首先根據(jù)題主的結(jié)果:

clipboard.png

a=1,b=2,c=function c(){}

其實代碼執(zhí)行完后,題主的代碼,就相當(dāng)于

var a=1; //這個地方也可能僅僅是 var a; a=1 的賦值操作還保留在下面函數(shù)內(nèi),不好驗證
var  b = function c () {
    b=2; //成功覆蓋了前面的 b,把 b 由function c 改成了 2;
    c=3; //函數(shù)體內(nèi)存在局部變量 c 指代函數(shù)本身,但是這個局部變量不可寫,所以賦值失敗
    console.log(a);
    console.log(b);
    console.log(c);
}
b();

主要是兩點:
1.局部變量定義 a=1 沒有用 var,所以js后臺會默認(rèn)你定義的是全局變量,所以題主的 a 是 1;

clipboard.png

2.全局變量 b 在函數(shù)內(nèi)被 b=2 重新賦值,所以輸出 2;
3、重點,需要舉例子

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

clipboard.png

這里返回的是函數(shù) c ,原因如下:

clipboard.png

那么 c=3; 其實就相當(dāng)于重新賦值了,那這種情況下,你要看下它允不允許重新賦值了,代碼:

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

clipboard.png

這就證明了這個賦值是失敗的,函數(shù)定義表達(dá)式帶名稱中的函數(shù)體內(nèi)的名稱標(biāo)識符指向的是函數(shù)本身,且不可寫,嚴(yán)格模式下,會報錯:

clipboard.png

字面理解就是賦值給了常量,所以如果你硬是要改變 c 的值,可以這樣聲明一個變量就好:

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

最后,這種函數(shù)定義表達(dá)式帶名稱的用法不常見,但是用來遞歸的時候很方便,詳情戳這里

2018年4月7日 14:35
編輯回答
任她鬧

簡單回答一下吧。

var b = function c()

函數(shù)的名字是 c 而不是 b。

先理解 2 個概念:函數(shù)表達(dá)式與函數(shù)聲明。

function c (){};

函數(shù)聲明,因為它是程序的一部分。

var b = function c(){};

函數(shù)表達(dá)式,因為它是賦值表達(dá)式(AssignmentExpression)的一部分。

兩者有一個差別就是:函數(shù)聲明會在任何表達(dá)式被解析和求值之前先行被解析和求值。即使聲明位于源代碼中的最后一行,它也會先于同一作用域中位于最前面的表達(dá)式被求值。

在你給出的代碼中:

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

執(zhí)行的結(jié)果是:

  • a 是 1
  • b 是 2
  • c 是函數(shù)

為什么 b 不是函數(shù)呢?而 c 卻是函數(shù)呢?

如果我們把最后的函數(shù)調(diào)用換成 c:

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

運(yùn)行時會報錯:

平時我們都是使用匿名函數(shù)表達(dá)式,突然出現(xiàn)一個 var b = function c(){} 時突然懵逼了,難怪 @shangguanyuanheng 會有疑問:“還有這種操作??”

函數(shù)有名字不是很正常的事兒嗎。但是這個函數(shù)名只能在函數(shù)內(nèi)部使用。So,第二個疑問解開了:

為什么調(diào)用 c() 時拋出了 Uncaught TypeError: c is not a function 異常,因為 c 只在函數(shù)內(nèi)部有效。

最后解釋題主的疑問:

在函數(shù)體里面,給 c 賦值,為什么會失???

前面加個 'use strict' 就清楚了。

大概查了一下規(guī)范,在具名函數(shù)表達(dá)式的規(guī)范中,可選的函數(shù)名(標(biāo)識 Identifier)會執(zhí)行 CreateImmutableBinding http://www.ecma-international...

2018年5月25日 08:12
編輯回答
淡墨

雖然講不出為什么
但是
這種代碼 可讀性極差
祝早日出現(xiàn)大神能解答一下

2018年6月1日 03:29
編輯回答
九年囚
var  b = function c () //還有這種操作??
2017年7月26日 03:51