2017-12-18 271 views
3

有人可能有任何想法如何將其替換爲stream()更改兩個foreach到一個流

for (Customer customer : customers) { 
    if (customer.getProducts() != null && customer.getProducts().getProduct() != null) { 
     for (Product product : customer.getProducts().getProduct()) { 
      if (product.getId().equals(productId)) { 
       return Optional.of(product); 
      } 
     } 
    } 
} 

我知道,第二的foreach我可以替換爲:

customer.getProducts().getProduct().stream().filter(a -> a.getId().equals(productId)).findAny(); 

但我怎麼能更換一個流都foreachs?

+1

查找到flatMap() –

+1

大概[該線程的重複](https://stackoverflow.com/questions/34406744/refactoring-a-nested-foreach-into-java-8-stream) – GuyKhmel

回答

4

使用flatMap得到所有Customer s的所有Product中的Stream

Optional<Product> product = 
    customers.stream() 
      .filter(c -> c.getProducts() != null && c.getProducts().getProduct() != null) 
      .flatMap(c -> c.getProducts().getProduct().stream()) 
      .filter(p -> p.getId().equals(productId)) 
      .findFirst(); 
2

就像@Eran的答案已經是正確的,我更喜歡下面的更多。唯一的區別是有每行只有一個操作(具有終於更線路的成本),這是在眼睛清潔:

Optional<Product> product = customers.stream()  // Stream<Customer> 
     .map(Customer::getProducts)    // Stream<Products> 
     .filter(Objects::nonNull)     // filter null values out 
     .map(Products::getProduct)    // Stream<Collection<Product>> 
     .filter(Objects::nonNull)     // filter null values out 
     .flatMap(Collection::stream)    // Stream<Product> 
     .filter(p -> p.getId().equals(productId)) // filter product with id out 
     .findAny();        // Optional<Product>