Skip to main content
 首页 » 编程设计

Java Optional API 新增方法

2022年07月19日117傻小

Java 9 Optional API 新增方法

本文介绍Java 9 Optional API 新增方法。除了模块化,Java 9 也给Optional类增加了三个方法。

1. or 方法

有时当Optional为空时,我们想执行一些其他逻辑并也返回Optional。在Java9之前Optional类仅有orElse()和orElseGet()方法,但两者都返回非包装值。

Java9引入or()方法当Optional为空时返回另一个Optional。如果Optional有定义值,则传入or方法的lambda不被执行:

@Test 
public void givenOptional_whenPresent_thenShouldTakeAValueFromIt() { 
    //given 
    String expected = "properValue"; 
    Optional<String> value = Optional.of(expected); 
    Optional<String> defaultValue = Optional.of("default"); 
  
    //when 
    Optional<String> result = value.or(() -> defaultValue); 
  
    //then 
    assertThat(result.get()).isEqualTo(expected); 
} 

另外当Optional为空时,返回值同样是defaultValue:

@Test 
public void givenOptional_whenEmpty_thenShouldTakeAValueFromOr() { 
    // given 
    String defaultString = "default"; 
    Optional<String> value = Optional.empty(); 
    Optional<String> defaultValue = Optional.of(defaultString); 
  
    // when 
    Optional<String> result = value.or(() -> defaultValue); 
  
    // then 
    assertThat(result.get()).isEqualTo(defaultString); 
} 

2. ifPresentOrElse 方法

假设有个Optional实例,通常我们需要对其包装的值执行特定业务,同时如果Optional实例为空时需增加一些度量用于记录或跟踪该事实。

ifPresentOrElse()方法就是为该场景而创建。我们可以传入一个Consumer用于执行Optional存在时,而Runnable用于在Optional为空时执行。

下面示例中,Optional存在,我们需要在值存在时增加特定计数器:

@Test 
public void givenOptional_whenPresent_thenShouldExecuteProperCallback() { 
    // given 
    Optional<String> value = Optional.of("properValue"); 
    AtomicInteger successCounter = new AtomicInteger(0); 
    AtomicInteger onEmptyOptionalCounter = new AtomicInteger(0); 
  
    // when 
    value.ifPresentOrElse( 
      v -> successCounter.incrementAndGet(),  
      onEmptyOptionalCounter::incrementAndGet); 
  
    // then 
    assertThat(successCounter.get()).isEqualTo(1); 
    assertThat(onEmptyOptionalCounter.get()).isEqualTo(0); 
} 

注意,传入的回调参数没有被执行,下面看如何Optional为空时,回调参数将会执行:

@Test 
public void givenOptional_whenNotPresent_thenShouldExecuteProperCallback() { 
    // given 
    Optional<String> value = Optional.empty(); 
    AtomicInteger successCounter = new AtomicInteger(0); 
    AtomicInteger onEmptyOptionalCounter = new AtomicInteger(0); 
  
    // when 
    value.ifPresentOrElse( 
      v -> successCounter.incrementAndGet(),  
      onEmptyOptionalCounter::incrementAndGet); 
  
    // then 
    assertThat(successCounter.get()).isEqualTo(0); 
    assertThat(onEmptyOptionalCounter.get()).isEqualTo(1); 
} 

3. stream方法

最后一个方法,Java 9 给Optional类增加的stream()方法。

Java有非常流畅、优雅的Stream Api,用于操作集合实现函数式编程概念。Java 9 在Optional类中引入stream()方法,让我们把Optional实例视为Stream。

加入我们定义了一个Optional,并执行它的stream()方法,则会创建一个元素的流,使得Stream的所有Api都可以使用:

@Test 
public void givenOptionalOfSome_whenToStream_thenShouldTreatItAsOneElementStream() { 
    // given 
    Optional<String> value = Optional.of("a"); 
  
    // when 
    List<String> collect = value.stream().map(String::toUpperCase).collect(Collectors.toList()); 
  
    // then 
    assertThat(collect).hasSameElementsAs(List.of("A")); 
} 

另外,Optional不存在时,调用stream方法返回空流:

@Test 
public void givenOptionalOfNone_whenToStream_thenShouldTreatItAsZeroElementStream() { 
    // given 
    Optional<String> value = Optional.empty(); 
  
    // when 
    List<String> collect = value.stream() 
      .map(String::toUpperCase) 
      .collect(Collectors.toList()); 
  
    // then 
    assertThat(collect).isEmpty(); 
} 

在空Stream上的操作没有任何效果,但因为steam方法,我们可以链接Optional api和Stream api,使得代码更优雅、流畅。

4. 总结

本文我们介绍了Java 9 Optional Api新增的三个方法。or方法在Optional为空时返回Optional对象。 ifPresentOrElse()在值存在时执行Consumer参数,反之执行另一个参数回调参数。最后是Optional的stream()方法提供流API实现链式操作。


本文参考链接:https://blog.csdn.net/neweastsun/article/details/103651299
阅读延展