我正在阅读有关Java流的资料,并在阅读过程中发现新的东西。我发现的一个新东西是peek()函数。几乎所有我读到的peek说它应该用来调试你的流。
如果我有一个流,其中每个帐户都有一个用户名,密码字段和login()和loggedIn()方法。
我还有
Consumer<Account> login = account -> account.login();
and
Predicate<Account> loggedIn = account -> account.loggedIn();
为什么会这么糟糕?
List<Account> accounts; //assume it's been setup
List<Account> loggedInAccount =
accounts.stream()
.peek(login)
.filter(loggedIn)
.collect(Collectors.toList());
现在,据我所知,这完全是它的目的。它;
获取一个帐户列表
尝试登录每个帐户
过滤掉任何未登录的帐户
将已登录的帐户收集到一个新列表中
这样做的坏处是什么?有什么理由不让我继续吗?最后,如果不是这个解决方案,那么会是什么?
它的原始版本使用.filter()方法如下所示;
.filter(account -> {
account.login();
return account.loggedIn();
})
关键是:
不要以一种意想不到的方式使用API,即使它实现了您的直接目标。这种方法在将来可能会失效,而且对于未来的维护者来说也不清楚。
将其分解为多个操作并没有害处,因为它们是不同的操作。以一种不明确和意外的方式使用API是有害的,如果在未来的Java版本中修改了这种特定的行为,可能会产生后果。
在此操作中使用forEach将使维护者清楚地知道帐户的每个元素都有预期的副作用,并且您正在执行一些可能会改变它的操作。
它也更传统,因为peek是一个中间操作,在终端操作运行之前不会对整个集合进行操作,但forEach确实是一个终端操作。通过这种方式,您可以围绕代码的行为和流程进行强有力的论证,而不是询问peek在此上下文中的行为是否与forEach相同。
accounts.forEach(a -> a.login());
List<Account> loggedInAccounts = accounts.stream()
.filter(Account::loggedIn)
.collect(Collectors.toList());
虽然我同意上面的大多数答案,但我有一个案例,使用peek似乎是最干净的方法。
与您的用例类似,假设您希望仅对活动帐户进行过滤,然后在这些帐户上执行登录。
accounts.stream()
.filter(Account::isActive)
.peek(login)
.collect(Collectors.toList());
Peek有助于避免冗余调用,同时不必迭代集合两次:
accounts.stream()
.filter(Account::isActive)
.map(account -> {
account.login();
return account;
})
.collect(Collectors.toList());