先上代码
public static void main(String[] args) {
final ArrayList<Object> list = new ArrayList<>();
System.out.println(list.stream().findFirst().map(x -> true).orElse(print("orElse")));
System.out.println(list.stream().findFirst().map(x -> true).orElseGet(() -> print("orElseGet")));
list.add(1);
System.out.println(list.stream().findFirst().map(x -> true).orElse(print("orElse")));
System.out.println(list.stream().findFirst().map(x -> true).orElseGet(() -> print("orElseGet")));
}
public static boolean print(String str) {
System.out.println("do execute:" + str);
return false;
}
执行结果
do execute:orElse
false
do execute:orElseGet
false
do execute:orElse
true
true
optional为空时
- orElse 会执行,返回orElse中的结果
- orElseGet 会执行,返回orElseGet中的结果
optional有值时
- orElse 会执行,返回optional的结果
- orElseGet 不会执行,返回optional的结果
结论
- 无论optional是否有值,orElse都会被执行
- 只有optional为空时,orElseGet才会被执行
tips
- 有个场景要注意。就是通过optional来操作数据库,如果查询到有数据,则update,没数据则执行insert
- 代码可能是这样
Optional.ofNullable(selectEntity(id)).map(update(entity)).orElse(insert(entity))
- 这时,如果方法上再有事物注解@transactional,那么很有可能会引起Mysql事物锁等待超时 Lock wait timeout exceeded; try restarting transaction
- 原因是:在同一事务内先后对同一条数据进行插入和更新操作。因为orElse是必会执行的
- 所以orElse中,有具体方法操作时,一定要记得使用orElseGet