本文主要是介绍内部类如果要引用外部类的变量,则该变量必须为final,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
今天程序中突然贵蹦出来一个这么错误:
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;
}
这篇关于内部类如果要引用外部类的变量,则该变量必须为final的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!