您的位置:首页 > 其它

内部类如果要引用外部类的变量,则该变量必须为final

2011-12-12 10:20 204 查看
今天程序中突然贵蹦出来一个这么错误:

Cannot refer to a non-final variable service inside an inner class defined in a different method

上网查最一下才知道:

内部类引用了一个非final的变量,

内部类如果要引用外部类的变量,则该变量必须为final,这是规定

(我觉得应该是生命周期的问题! 如果你引用的外部变量不是一个final 的! 那么,可能在你的内部类运行期间 引用的外部变量 超出了生命周期之外(方法执行完了 ,可是内部类对象却还在引用该局部变量!))

涉及到闭包(closure)的概念。

不同语言对闭包选择了不同的支持,相比之下,C#中就能够在匿名函数中引用非final的外部变量,例如这篇帖子中提到的C#代码:

List<Func<int>> actions = new List<Func<int>>();

int variable = 0;
while (variable < 5)
{
actions.Add(() => variable * 2);
++ variable;
}

foreach (var act in actions)
{
Console.WriteLine(act.Invoke());
}

这一行:

actions.Add(() => variable * 2);

就在匿名方法内引用了外部局部变量: variable.

但同时也引起了一个奇怪的问题,上面的C#代码,我们期望的结果应该是输出"0, 2, 4, 6, 8"才对,结果却是10个零。

原因就是C# compiler 在所闭的包内圈入了对variable的引用,然后当运行

Console.WriteLine(act.Invoke());


时,variable值已经变成5,结果输出为10个零。

这种特性多少有点令人意外,可能java正是为了避免这种情形,于是强制规定闭包内引用的外部局部变量必须为final,想象一下如果上例中variable 为final的话,自然就不会写出上面的代码,造成类似的意外了。

另外,上例中如果你希望达到输出"0, 2, 4, 6, 8"的效果,可以:

while (variable < 5)
{
int copy = variable;
actions.Add(() => copy * 2);
++ variable;
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: 
相关文章推荐