据我了解,lambda表达式捕获值,而不是变量。例如,以下是编译时错误:
for (int k = 0; k < 10; k++) {
new Thread(() -> System.out.println(k)).start();
// Error—cannot capture k
// Local variable k defined in an enclosing scope must be final or effectively final
}
但是,当我尝试使用增强功能运行相同的逻辑时,for-loop
一切工作正常:
List<Integer> listOfInt = new Arrays.asList(1, 2, 3);
for (Integer arg : listOfInt) {
new Thread(() -> System.out.println(arg)).start();
// OK to capture 'arg'
}
为什么它对于增强型for
循环而不是常规的常规for
循环都能很好地工作,尽管增强型for
循环也像常规循环那样在内部递增变量。**
Lambda表达式的工作方式类似于回调。一旦在代码中传递它们,它们就“存储”它们需要操作的任何外部值(或引用)(就像这些值在函数调用中作为参数传递一样。这对开发人员来说是隐藏的)。在第一个示例中,可以通过存储k
到单独的变量(例如d)来解决该问题:
for (int k = 0; k < 10; k++) {
final int d = k
new Thread(() -> System.out.println(d)).start();
}
有效地final
意味着,在上面的示例中,您可以将’final’关键字排除在外,因为d
它实际上是最终的,因为它在其范围内从未更改。
for
循环的运行方式不同。它们是迭代代码(与回调相反)。它们在各自的范围内工作,并且可以在自己的堆栈上使用所有变量。这意味着for
循环的代码块是外部代码块的一部分。
关于您突出显示的问题:
增强的for
循环不能使用常规索引计数器运行,至少不能直接运行。增强的for
循环(在非数组上)创建一个隐藏的迭代器。您可以通过以下方式进行测试:
Collection<String> mySet = new HashSet<>();
mySet.addAll(Arrays.asList("A", "B", "C"));
for (String myString : mySet) {
if (myString.equals("B")) {
mySet.remove(myString);
}
}
上面的示例将导致ConcurrentModificationException。这是由于迭代器注意到基础集合在执行期间已更改。但是,在您的示例中,外部循环创建了一个“有效最终”变量arg
,该变量可以在lambda表达式中引用,因为该值是在执行时捕获的。
防止捕获“非有效最终”值或多或少只是Java中的一项预防措施,因为在其他语言(例如JavaScript)中,这是不同的。
因此,编译器理论上可以翻译代码,捕获值并继续,但是它必须以不同的方式存储该值,并且您可能会得到意想不到的结果。因此,为Java
8开发lambda的团队通过例外地防止了这种情况,从而正确排除了这种情况。
如果您需要更改lambda表达式中的外部变量的值,则可以声明一个单元素数组:
String[] myStringRef = { "before" };
someCallingMethod(() -> myStringRef[0] = "after" );
System.out.println(myStringRef[0]);
或用于AtomicReference<T>
使其成为线程安全的。但是,在您的示例中,此回调可能会返回“
before”,因为回调很可能在println执行之后执行。
问题内容: 我正在从Java切换到C ,并且想知道C 是否包含我在Java中使用的增强的for循环,例如: 在C ++中是否可能有相同的“快捷方式”? 问题答案: 在C ++ 11中,如果编译器支持,则可以。这称为基于范围的。 它适用于C样式数组以及具有函数并返回迭代器的任何类型。例:
问题内容: 在玩循环的同时创建了以下代码。下面的代码将斐波那契值存储到数组中,然后使用for循环将其打印出来。 上面的代码工作正常。但是,第一次将它们放在一起时,我使用了增强的for循环来打印出值(代码中的第二个for循环)。可以很好地编译,但是运行时得到以下信息: 我不明白出了什么问题。更改第二个循环不应更改值(您会注意到斐波那契值是错误的(即缺少值))。而且我不明白为什么简单的增强型for循环
问题内容: 我正在尝试找出部分任务,而现在我已经将头撞墙了。我正在尝试将DNA序列转录为RNA序列。但是,我正在获取ArrayOutOfBoundsException。我是使用增强的for循环进行迭代的新手,所以我的错误可能隐藏在某个地方。在满足if语句参数之前,它不会发生。 问题答案: 问题出在声明中 问题是内部表示为等于84,因此您得到一个。您需要使用传统计数器对其进行迭代:
问题内容: 这是我的代码: 增强的循环提供: 虽然此循环语句有效。为什么?代码有什么问题? 问题答案: 在这种情况下,将分配给数组中的每个元素-它 不是 数组的索引。 您想要做的是: 在您的代码中,您试图在迭代对象引用的数组索引处选择整数。换句话说,您的代码等效于:
我是Java的初学者,我有这个疑问。是否可以在ArrayList上使用Java中增强的for循环,但从指定的点而不是ArrayList开始[0]。 我可以使用增强的for循环并从calc[2]而不是calc[0]开始迭代吗??如果可能,我该怎么做?在我的特定情况下,使用增强的for循环会更好,而不是普通的for循环。
问题内容: 在Java中的for循环中防止空值的最佳方法是什么? 这看起来很丑: 要么 可能没有其他办法。他们是否应该将它放在构造本身中,如果它为null,则不要运行循环? 问题答案: 您最好验证从哪里获得该列表。 空列表就是您所需要的,因为空列表不会失败。 如果您从其他地方获得此列表,并且不知道是否可以,则可以创建一个实用程序方法并像这样使用它: 当然是: