匿名函數是一個“內聯”語句或表達式,可在需要委託類型的任何地方使用。
可以使用匿名函數來初始化命名委託[無需取名字的委託],或傳遞命名委託(而不是命名委託類型,傳遞一個方法塊,而不是委託類型)[callback的方式]作爲方法參數。
兩種匿名函數#
共有兩種匿名函數,以下主題分別討論了這些函數
- Lambda表達式
- 匿名方法
匿名函數 -(C# 編程指南)
匿名函數是一個“內聯”語句或表達式,可在需要委託類型的任何地方使用。 可以使用匿名函數來初始化命名委託,或傳遞命名委託(而不是命名委託類型)作爲方法參數。
可以使用 lambda 表達式或匿名方法來創建匿名函數。 建議使用 lambda 表達式,因爲它們提供了更簡潔和富有表現力的方式來編寫內聯代碼。 與匿名方法不同,某些類型的 lambda 表達式可以轉換爲表達式樹類型。
C# 中委託的演變
在 C# 1.0 中,通過使用在代碼中其他位置定義的方法顯式初始化委託來創建委託的實例。 C# 2.0 引入了匿名方法的概念,作爲一種編寫可在委託調用中執行的未命名內聯語句塊的方式。 C# 3.0 引入了 lambda 表達式,這種表達式與匿名方法的概念類似,但更具表現力並且更簡練。 這兩個功能統稱爲匿名函數。 通常,面向 .NET Framework 3.5 或更高版本的應用程序應使用 lambda 表達式。
下面的示例演示從 C# 1.0 到 C# 3.0 委託創建過程的發展:
static void Main(string[] args) { // Original delegate syntax required // initialization with a named method. TestDelegate testDelA = new TestDelegate(M); // C# 2.0: A delegate can be initialized with // inline code, called an "anonymous method." This // method takes a string as an input parameter. TestDelegate testDelB = delegate(string s) { Console.WriteLine(s); }; // C# 3.0. A delegate can be initialized with // a lambda expression. The lambda also takes a string // as an input parameter (x). The type of x is inferred by the compiler. TestDelegate testDelC = (x) => { Console.WriteLine(x); }; // Invoke the delegates. testDelA("Hello. My name is M and I write lines."); testDelB("That's nothing. I'm anonymous and "); testDelC("I'm a famous author."); // Keep console window open in debug mode. Console.WriteLine("Press any key to exit."); Console.ReadKey(); }
匿名函數顧名思義指的是沒有名字的函數,在實際開發中使用的頻率非常高!也是學好JS的重點。
匿名函數:沒有實際名字的函數。
首先我們聲明一個普通函數:
//聲明一個普通函數,函數的名字叫fn
function fn(){
console.log("張培躍");
}
然後將函數的名字去掉即是匿名函數:
//匿名函數,咦,運行時,你會發現報錯啦!
function (){
console.log("張培躍");
}
到此,你會發現單獨運行一個匿名函數,由於不符合語法要求,報錯啦!解決方法只需要給匿名函數包裹一個括號即可:
//匿名函數在其它應用場景括號可以省略
(function (){
//由於沒有執行該匿名函數,所以不會執行匿名函數體內的語句。
console.log("張培躍");
})
如果需要執行匿名函數,在匿名函數後面加上一個括號即可立即執行!
(function (){
//此時會輸出張培躍
console.log("張培躍");
})()
倘若需要傳值,直接將參數寫到括號內即可:
(function (str){
//此時會輸出張培躍好帥!
console.log("張培躍"+str);
})("好帥!")
匿名函數的應用場景
1、事件
<input type="button" value="點我啊!" id="sub">
<script>
//獲得按鈕元素
var sub=document.querySelector("#sub");
//給按鈕增加點擊事件。
sub.onclick=function(){
alert("當點擊按鈕時會執行到我哦!");
}
</script>
2、對象
var obj={
name:"張培躍",
age:18,
fn:function(){
return "我叫"+this.name+"今年"+this.age+"歲了!";
}
};
console.log(obj.fn());//我叫張培躍今年18歲了!
3、函數表達式
//將匿名函數賦值給變量fn。
var fn=function(){
return "我是一隻小小小小留下,怎麼飛也飛不高!"
}
//調用方式與調用普通函數一樣
console.log(fn());//我是一隻小小小小留下,怎麼飛也飛不高!
4、回調函數
setInterval(function(){
console.log("我其實是一個回調函數,每次1秒鐘會被執行一次");
},1000);
5、返回值
//將匿名函數作爲返回值
function fn(){
//返回匿名函數
return function(){
return "張培躍";
}
}
//調用匿名函數
console.log(fn()());//張培躍
//或
var box=fn();
console.log(box());//張培躍
模仿塊級作用域
塊級作用域,有的地方稱爲私有作用域。JavaScript中是沒有塊級作用域的,例如:
if(1==1){//條件成立,執行if代碼塊語句。
var a=12;//a爲全局變量
}
console.log(a);//12
for(var i=0;i<3;i++){
console.log(i);
}
console.log(i);//4
if(){}for(){}等沒有自己的作用域。如果有,出了自己的作用域,聲明的變量就會立即被銷燬了。但是咱們可以通過匿名函數來模擬塊級作用域:
(function(){
//這裏是我們的塊級作用域(私有作用域)
})();
嘗試塊級作用域:
function fn(){
(function(){
var la="啦啦啦!";
})();
console.log(la);//報錯---la is not defined
}
fn();
匿名函數的作用:
1、通過匿名函數可以實現閉包,關於閉包在後面的文章中會重點講解。在這裏簡單介紹一下:閉包是可以訪問在函數作用域內定義的變量的函數。若要創建一個閉包,往往都需要用到匿名函數。
2、模擬塊級作用域,減少全局變量。執行完匿名函數,存儲在內存中相對應的變量會被銷燬,從而節省內存。再者,在大型多人開發的項目中,使用塊級作用域,會大大降低命名衝突的問題,從而避免產生災難性的後果。自此開發者再也不必擔心搞亂全局作用域了。