sum захватывается анонимным методом. Метод Counter () вызывается в методе Main () для получения объекта Countlt, а следовательно, переменная sum не уничтожается до самого конца программы.
// Продемонстрировать применение захваченной переменной, using System;
// Этот делегат возвращает значение типа int и принимает аргумент типа int. delegate int Countlt(int end);
class VarCapture {
static Countlt Counter () {
int sum = 0;
// Здесь подсчитанная сумма сохраняется в переменной sum.
Countlt ctObj = delegate (int end) { for(int i=0; i <= end; i++) {
Console.WriteLine(i); sum += i;
}
return sum;
};
return ctObj;
}
static void Main() {
// Получить результат подсчета.
Countlt count = Counter ();
int result;
result = count(3);
Console.WriteLine('Сумма 3 равна ' + result);
Console.WriteLine();
result = count(5);
Console.WriteLine('Сумма 5 равна ' + result);
}
}
Ниже приведен результат выполнения этой программы. Обратите особое внимание на суммарное значение.
0
3
Сумма 3 равна 6
0
1
2
3
4
5
Сумма 5 равна 21
Как видите, подсчет по-прежнему выполняется как обычно. Но обратите внимание на то, что сумма 5 теперь равна 21, а не 15! Дело в том, что переменная sum захватывается объектом ctOb j при его создании в методе Counter (). Это означает, что она продолжает существовать вплоть до уничтожения делегата count при 'сборке мусо-ра' в самом конце программы. Следовательно, ее значение не уничтожается после возврата из метода Counter () или при каждом вызове анонимного метода, когда происходит обращение к делегату count в методе Main ().
Несмотря на то что применение захваченных переменных может привести к довольно неожиданным результатам, как в приведенном выше примере, оно все же логически обоснованно. Ведь когда анонимный метод захватывает переменную, она продолжает существовать до тех пор, пока используется захватывающий ее делегат. В противном случае захваченная переменная оказалась бы неопределенной, когда она могла бы потребоваться делегату.
Лямбда-выражения
