C#でデリゲートやラムダ式など使い、いろいろ遊んでみました
program.cs
usingSystem;namespaceCsLamTest{classProgram{staticvoidMain(string[]args){Func<int>addDelegate=(//(1)()=>{//(1-1)Console.WriteLine("初期化の関数から出力");//(1-2)return0;});f(//(2)()=>{//(2-1)Console.WriteLine("ラムダ式から出力");//(2-2)return5*5;},refaddDelegate//(2-3));f(//(3)delegate(){//(3-1)Console.WriteLine("デリゲートから出力");//(3-2)return3*3;},refaddDelegate//(3-3));}staticvoidf(Func<int>func,refFunc<int>addDelegate)//(4){addDelegate+=func;//(4-1)Console.WriteLine(addDelegate());//(4-2)}}}出力結果は
初期化の関数から出力
ラムダ式から出力
25
初期化の関数から出力
ラムダ式から出力
デリゲートから出力
9
解説
まずは(1)の部分
program.cs
Func<int>addDelegate=(//(1)()=>{//(1-1)Console.WriteLine("初期化の関数から出力");//(1-2)return0;});(1)の行は、Func<int>(引数を取らずint型を返す関数)型のaddDelegateを定義しています。addDelegateの初期化が(1-1)でラムダ式を用いて行われています。
次に(2)の部分
program.cs
f(//(2)()=>{//(2-1)Console.WriteLine("ラムダ式から出力");//(2-2)return5*5;},refaddDelegate//(2-3));(2)で(4)の関数fを呼び出しています。
その第1引数Func<int> funcに(2-1)で示されたラムダ式を渡しています。
第2引数ref Func<int> addDelegateに(1)のaddDelegateを渡しています。・・・(2-3)の行。
(3)はラムダ式ではなく無名関数を渡しています。他は(2)と同じです。
次に(4)の部分
program.cs
staticvoidf(Func<int>func,refFunc<int>addDelegate)//(4){addDelegate+=func;//(4-1)Console.WriteLine(addDelegate());//(4-2)}(4-1)で、第1引数で渡されたfuncを、第2引数で渡されたaddDelegateに加えています。
デリゲートは、+=演算子を用いて、関数を連結できます。
連結された関数は、古いものから順番に実行されます。
実行の流れ
- まず
(2)で関数fを呼び出し (4-1)でaddDelegateに(2-1)のラムダ式を加える(4-2)でaddDelegate()を実行(1-2)が出力される(2-2)が出力される(2-1)のラムダ式の戻り値が出力される。
次に
(3)で関数fを呼び出し(4-1)でaddDelegateに(3-1)のデリゲートを加える3.と同じ
(1-2)(2-2)(3-2)が順番に出力される(3-1)のデリゲートの戻り値が出力される
今回のC#いじりで知ったこと
Func<int>には、ラムダ式も無名関数も渡せる、代入できる。Console.WriteLine()では、デリゲートに複数の関数が+=で連結されていても、最後に実行される関数の戻り値のみ表示する。
デリゲートは値型だった。なので、refをつけています。