JavaScript 基礎(chǔ)函數(shù)_深入剖析變量和作用域
來源:易賢網(wǎng) 閱讀:768 次 日期:2016-06-27 14:11:54
溫馨提示:易賢網(wǎng)小編為您整理了“JavaScript 基礎(chǔ)函數(shù)_深入剖析變量和作用域”,方便廣大網(wǎng)友查閱!

下面小編就為大家?guī)硪黄狫avaScript 基礎(chǔ)函數(shù)_深入剖析變量和作用域。小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。

函數(shù)定義和調(diào)用

定義函數(shù),在JavaScript中,定義函數(shù)的方式如下:

function abs(x){

if(x >=0){

return x;

}else{

return -x;

}

}

上述abs() 函數(shù)的定義如下:

function 指出這是一個函數(shù)定義;

abs 是函數(shù)的名稱;

(x) 括號內(nèi)列出函數(shù)的參數(shù),多個參數(shù)以,分隔;

{...}之間的代碼是函數(shù)體,可以包含若干語句,甚至可以沒有任何語句。

注意:函數(shù)體內(nèi)部的語句在執(zhí)行時,一旦執(zhí)行到return 時,函數(shù)就執(zhí)行完畢,并將結(jié)果返回。因此內(nèi)部通過條件判斷和循環(huán)可以在實現(xiàn)非常復(fù)雜的。

如果沒有return語句,函數(shù)執(zhí)行完畢后也會返回結(jié)果,只是結(jié)果為undefined。

由于JavaScript的函數(shù)也是一個對象,上述定義的abs()函數(shù)實際上是一個函數(shù)對象,而函數(shù)名abs可以視為指向該函數(shù)的變量。

var abs = function(x){

if(x >= 0){

return x;

} else {

return -x;

}

}

在這種方式下,function (x) { ... }是一個匿名函數(shù),它沒有函數(shù)名。但是,這個匿名函數(shù)賦值給了變量abs,所以,通過變量abs就可以調(diào)用該函數(shù)。

兩種定義完全等價,注意第二種方式按照完整語法需要在函數(shù)體末尾加一個;,表示賦值語句結(jié)束。

調(diào)用函數(shù)時,按順序傳入?yún)?shù)即可:

abs(10); // 返回10

abs(-9); // 返回9

由于JavaScript 允許傳入任意個參數(shù)而不受影響調(diào)用,因此傳入的參數(shù)比定義的參數(shù)多也沒有問題,雖然函數(shù)內(nèi)部并不需要這些參數(shù)。

abs(10,'blablabla'); //返回10

abs(-9,'haha','hehe',null) // 返回9

傳入的參數(shù)比定義的少也沒有問題

abs(); 返回NaN

此時abs(x)函數(shù)的參數(shù)x 將收到undefined,計算結(jié)果為NaN

function abs(x){

if(typeof x !=='number'){

throw 'Not a number':

}

if(x >=0){

return x;

}else{

return -x;

}

}

arguments

JavaScript 還有一個免費贈送的關(guān)鍵字 arguments,它只在函數(shù)內(nèi)部起作用,并且永遠指向當前函數(shù)的調(diào)用者傳入的所有參數(shù)。

function foo(x){

alert(x); // 10

for(var i=0; i < arguments.length;++){

alert(arguments[i]); // 10,20,30

}

}

foo(10.20,30)

利用arguments,你可以獲得調(diào)用者傳入的所有參數(shù)。也就是說,即使函數(shù)不定義任何參數(shù),還是可以拿到參數(shù)的值:

function abs(){

if(arguments.length ===0){

return 0;

}

var x = arguments[0]

return x >=0 ? x : -x;

}

abs(); //0

abs(10); // 10

abs(-9) //9

實際上arguments最常用于判斷傳入?yún)?shù)的個數(shù)。你可能會看到這樣的寫法:

// foo(a[,b],c)

//接受2~3 個參數(shù),b 是可選參數(shù),如果只要出入兩個參數(shù),b默認為null

function foo(a,b,c){

if(arguments.length ===2){

// 實際拿到的參數(shù)是a 和b c 為undefined

c = b;

b = null; // b 變?yōu)槟J值

要把中間的參數(shù)b變?yōu)椤翱蛇x”參數(shù),就只能通過arguments判斷,然后重新調(diào)整參數(shù)并賦值。

rest 參數(shù)

由于JavaScript 函數(shù)允許接收任意個參數(shù),遇事我們就不得不用arguments 來獲取所有的參數(shù):

function foo(a,b){

var i, rest = [];

if(arguments.length > 2){

for(i = 2; i < arguments.length; i++){

rest.push(arguments[i]);

}

}

console.log('a =' + a);

console.log('b = ' + b);

console.log(rest);

}

為了獲取除了已定義參數(shù)a、b之外的參數(shù),我們不得不用arguments,并且循環(huán)要從索引2開始以便排除前兩個參數(shù),這種寫法很別扭,只是為了獲得額外的rest參數(shù),有沒 有更好的方法?

ES6標準引入了rest參數(shù),上面的函數(shù)可以改寫為:

function foo(a,b,...rest){

console.log('a = ' + a);

console.log('b = ' + b);

console.log(rest);

}

foo(1,2,3,4,5);

//結(jié)果

// a = 1

// b = 2

// Array[3,4,5]

foo(1)

// 結(jié)果

// a = 1

// b = undefined

// Array []

rest 參數(shù)只能寫在最后,前面用... 標示,從運行結(jié)果可知,傳入的參數(shù)先綁定 a , b, 多余的參數(shù)以數(shù)組形式交給變量 rest,所以,

不在需要 arguments 我們就獲取了全部參數(shù)。

如果傳入的參數(shù)連正常定義的參數(shù)都沒填滿,也不要緊,rest參數(shù)會接收一個空數(shù)組(注意不是undefined)。

return 語句

前面我們講到了JavaScript引擎有一個在行末自動添加分號的機制,這可能讓你栽到return語句的一個大坑:、

function foo(){

return {name:'foo'};

}

foo(); // {name:'foo'}

要注意:

function foo(){

return: //自動添加了分號,相當于return undefined

{name:'foo'}; // 這行語句已經(jīng)沒法執(zhí)行到了。

}

所以正確的多行寫法是

function foo(){

return { // 這里不會自動加分號,因為表示語句尚未結(jié)束。

name:'foo'

}

}

變量作用域

在JavaScript 中,用var 聲明的實際上是有作用域的。

如果一個變量在函數(shù)體內(nèi)部申明,則該變量的作用域為整個函數(shù)體,在函數(shù)體外不該引用該變量。

‘use strict':

function foo(){

var x = 1;

x = x +1;

}

x = x +2; // RefrenceError 無法在函數(shù)體外引用該該變量x

如果兩個不同的函數(shù)各自申明了同一個變量,那么該變量只在各自的函數(shù)體內(nèi)起作用。換句話說,不同函數(shù)內(nèi)部的同名變量互相獨立,互不影響:

'use struct':

function foo(){

var x = 1;

x = x +1;

}

function bar (){

var x= 'A';

x = x + 'B';

}

由于JavaScript的函數(shù)可以嵌套,此時,內(nèi)部函數(shù)可以訪問外部函數(shù)定義的變量,反過來則不行:

'use strict';

function foo(){

var x =1;

function bar(){

var x = 1;

function bar(){

var y= x +1; //bar 可以訪問foo 的變量x

}

var z = y + 1; //RefernceError! foo 不可以訪問bar 的變量y!

}

}

如果內(nèi)部函數(shù)和外部函數(shù)的變量名重名怎么辦?

'use strict':

function foo(){

var x = 1;

function bar (){

var x = 'A';

alert('x in bar() =' + x); // 'A'

}

alert('x in foo()=' +x) //1

bar();

}

變量提升

JavaScript的函數(shù)定義有個特點,它會先掃描整個函數(shù)體的語句,把所有申明的變量“提升”到函數(shù)頂部:

'use strict';

function foo(){

var x='Hello,'+y;

alert(x);

var y = 'Bob';

}

foo();

對于上述foo()函數(shù),JavaScript引擎看到的代碼相當于:

function foo(){

var y; // 提升變量y的

var x = 'Hello' + y;

alert(x);

y = 'Bob';

}

由于JavaScript的這一怪異的“特性”,我們在函數(shù)內(nèi)部定義變量時,請嚴格遵守“在函數(shù)內(nèi)部首先申明所有變量”這一規(guī)則。最常見的做法是用一個var申明函數(shù)內(nèi)部用到的所有變量:

function foo(){

var x =1, // x 初始化為1

y = x +1, // y 初始化為2

z,i; // z和i 為undefined

// 其他語句

for(i =0; i<100; i++){

...

}

}

全局作用域

不在任何函數(shù)內(nèi)定義的變量就具有全局作用域,實際上,JavaScript 默認有一個全局作用域的變量實際上唄綁定到window 的一個屬性。

‘use strict';

var sourse = 'Learn JavaScript';

alert(course); // 'Learn JavaScript';

alert(window.course); // 'Learn JavaScript'

名字空間

全局變量會綁定到window 上,不同的JavaScript 文件如果使用相同的全局變量,或者定義了相同名字的頂層函數(shù),都會造成

命名沖突,并且很難被發(fā)現(xiàn),

減少沖突的一個方法是把自己的所有的變量和函數(shù)全部綁定到一個全局變量中。

// 唯一的曲劇變量MYAPP

var MYAPP = {};

//其他變量:

MYAPP.name = 'myapp';

MYAPP.version = 1.0;

// 其他函數(shù)

MYAPP.foo = function (){

return 'foo';

};

把自己的代碼全部放入唯一的名字空間MYAPP中,會大大減少全局變量沖突的可能。

局部作用域

由于JavaScript 的變量作用域?qū)嶋H上是函數(shù)內(nèi)部,我們在for 循環(huán)等語句塊中是無法定義具有無法定義具有局部作用域的變量的。

function foo(){

for(var i = 0; i<100; i++){

//

}

i+=100; // 仍然可以引用變量;

}

為了解決塊級作用域,ES6引入了新的關(guān)鍵字let,用let替代var可以申明一個塊級作用域的變量:

function foo(){

var sum = 0;

for(let i=0; i<100;i++){

sum +=i;

}

i +=1;

}

常量

由于var 和let 聲明的變量,如果要聲明一個常量,在ES6 之前是不行的,我們通常用全部大寫的變量倆表示這是一個常量

不要修改他的值。

var PI = 3.14;

ES6標準引入了新的關(guān)鍵字const 來定義常量,const 與 let都具有塊級作用域;

const PI = 3.14;

PI = 3; // 某些瀏覽器不報錯,但是無效果。

PI; // 3.14

以上這篇JavaScript 基礎(chǔ)函數(shù)_深入剖析變量和作用域就是小編分享給大家的全部內(nèi)容了,希望能給大家一個參考

更多信息請查看網(wǎng)絡(luò)編程
易賢網(wǎng)手機網(wǎng)站地址:JavaScript 基礎(chǔ)函數(shù)_深入剖析變量和作用域

2025國考·省考課程試聽報名

  • 報班類型
  • 姓名
  • 手機號
  • 驗證碼
關(guān)于我們 | 聯(lián)系我們 | 人才招聘 | 網(wǎng)站聲明 | 網(wǎng)站幫助 | 非正式的簡要咨詢 | 簡要咨詢須知 | 新媒體/短視頻平臺 | 手機站點 | 投訴建議
工業(yè)和信息化部備案號:滇ICP備2023014141號-1 云南省教育廳備案號:云教ICP備0901021 滇公網(wǎng)安備53010202001879號 人力資源服務(wù)許可證:(云)人服證字(2023)第0102001523號
聯(lián)系電話:0871-65099533/13759567129 獲取招聘考試信息及咨詢關(guān)注公眾號:hfpxwx
咨詢QQ:1093837350(9:00—18:00)版權(quán)所有:易賢網(wǎng)