zhoumy博客(C#、Windows Phone XAML)

C#:在匿名方法中捕获外部变量

先来一段代码引入主题。如果你可以直接说出代码的输出结果,说明本文不适合你。(代码引自《深入理解C#》第三版)

    class Program
    {
        private delegate void TestDelegate();
 
        static void Main(string[] args)
        {
            TestDelegate[] delegates = new TestDelegate[2];

            int outside = 0;

            for(int i = 0; i < 2; i++)
            {
                int inside = 0;

                delegates[i] = delegate
                {
                    Console.WriteLine("({0},{1})", outside, inside);
                    outside++;
                    inside++;
                };
            }

            delegates[0]();
            delegates[0]();
            delegates[0]();

            delegates[1]();
            delegates[1]();

            Console.ReadKey();
        }
    }

答案在本文最下方。

先引入两个定义:

1、外部变量:是指作用域内包括匿名方法的局部变量或参数。

2、捕获的外部变量:是指在匿名方法中使用的外部变量。

定义有点抽象,针对上面的代码而言,inside和outside都是匿名方法的外部变量,而inside和outside同样也作为被匿名方法捕获的外部变量,因为在匿名方法体中引用了这两个变量。

而被匿名方法捕获到的是变量本身,而并非变量的值。针对上面的代码而言,执行outside++和inside++时,操作的其实就是外部的inside和outside变量。

至此,我们应该可以得出outside的变化应该是从0到4。

那么程序的输出也应该是这个样子的:(X代码未知)

(0,X)

(1,X)

(2,X)

(3,X)

(4,X)

 

实际上,在for循环体里面做的东西就是实例化了两个TestDelegate委托对象:delegate[0]和delegate[1]

而循环的执行,实际上会实例化两个名称都为inside的变量。

这两个变量之间没有什么关联,仅仅是名称一样而已。

这两个变量分别作为delegate[0]和delegate[1]的外部捕获变量。也就是delegate[0]和delegate[1]在操作inside时,是互不影响的。

于是,得出答案:

0,0)

(1,1)

(2,2)

(3,0)

(4,1
答案

 

posted @ 2015-03-21 16:09  zhoumy  阅读(2127)  评论(1编辑  收藏  举报
zhoumy博客(C#、Windows Phone XAML)