Захват локальной переменной может привести к неожиданным результатам. В качестве примера рассмотрим еще один вариант программы подсчета с суммированием чисел. В данном варианте объектCountltконструируется и возвращается статическим методомCounter (). Этот объект использует переменнуюsum,объявленную в охватывающей области действия методаCounter (), а не самого анонимного метода. Поэтому переменная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

1

2

3

Сумма 3 равна 6

0

1

2

3

4

5

Сумма 5 равна 21

Как видите, подсчет по-прежнему выполняется как обычно. Но обратите внимание на то, что сумма 5 теперь равна 21, а не 15! Дело в том, что переменнаяsumзахватывается объектомctOb jпри его создании в методеCounter (). Это означает, что она продолжает существовать вплоть до уничтожения делегатаcountпри "сборке мусо-ра" в самом конце программы. Следовательно, ее значение не уничтожается после возврата из методаCounter() или при каждом вызове анонимного метода, когда происходит обращение к делегатуcountв методеMain().

Несмотря на то что применение захваченных переменных может привести к довольно неожиданным результатам, как в приведенном выше примере, оно все же логически обоснованно. Ведь когда анонимный метод захватывает переменную, она продолжает существовать до тех пор, пока используется захватывающий ее делегат. В противном случае захваченная переменная оказалась бы неопределенной, когда она могла бы потребоваться делегату.

Лямбда-выражения

Перейти на страницу:

Похожие книги