下面小編就為大家?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)容了,希望能給大家一個參考