溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Java集合Stream流如何使用

發(fā)布時(shí)間:2023-02-22 16:53:49 來源:億速云 閱讀:129 作者:iii 欄目:開發(fā)技術(shù)

這篇文章主要介紹“Java集合Stream流如何使用”的相關(guān)知識(shí),小編通過實(shí)際案例向大家展示操作過程,操作方法簡單快捷,實(shí)用性強(qiáng),希望這篇“Java集合Stream流如何使用”文章能幫助大家解決問題。

Java 中可以使用 java.util.Stream 對一個(gè)集合(實(shí)現(xiàn)了java.util.Collection接口的類)做各種操作,例如:求和、過濾、排序等等。

這些操作可能是中間操作——返回一個(gè) Stream 流,或者是終端操作——返回一個(gè)結(jié)果。

流操作并不會(huì)影響原來的集合,可以簡單認(rèn)為,流操作是把集合中的一個(gè)元素逐個(gè)復(fù)制放到一個(gè)首尾相接的流動(dòng)的水槽中。

Stream 流支持同步執(zhí)行,也支持并發(fā)執(zhí)行。如果我們直接獲取 stream 流,得到的是同步執(zhí)行的 stream 流;如果調(diào)用方法 parallelStream,則得到一個(gè)可以并發(fā)執(zhí)行的 Stream 流。

注意:Map不支持 Stream 流,但是它的 KeyValue 支持,因?yàn)樗鼈儗?shí)現(xiàn)了 Set 接口。

事前準(zhǔn)備

演示 Stream 流的提前準(zhǔn)備,創(chuàng)建幾個(gè)類以供測試

  • 新建一個(gè)工具類,方便創(chuàng)建集合。

  • 新建兩個(gè)類,例如開發(fā)中常見的數(shù)據(jù)庫實(shí)體類和 DTO 類。

 public class MyUtil {
 
     private static List<String> list = new ArrayList<>();
     private static List<Student> students = new ArrayList<>();
 
     static {
         list.add("abc");
         list.add("xyz");
         list.add("fgh");
         list.add("abc");
         list.add("def");
         list.add("xyz");
         list.add("efg");
 
         Student s1 = new Student();
         s1.setAge("16");
         s1.setId(UUID.randomUUID().toString());
         s1.setName("張三");
         s1.setMajor("計(jì)算機(jī)科學(xué)與技術(shù)");
         Student s2 = new Student();
         s2.setAge("18");
         s2.setId(UUID.randomUUID().toString());
         s2.setName("李四");
         s2.setMajor("物聯(lián)網(wǎng)工程");
         Student s3 = new Student();
         s3.setAge("20");
         s3.setId(UUID.randomUUID().toString());
         s3.setName("王五");
         s3.setMajor("網(wǎng)絡(luò)工程");
         students.add(s1);
         students.add(s2);
         students.add(s3);
     }
 
     public static List<String> getList() {
         return list;
     }
     public static List<Student> getStudents() {
         return students;
     }
 }
 
 public class Student {
 
     private String id;
     private String name;
     private String age;
     private String major;
     
 }
 
 public class StudentDTO {
 
     private String name;
     private String major;
 }

Filter

filter 可以幫助我們過濾流中的某些元素,其方法簽名如下

 /*
 過濾操作,
 Predicate 相當(dāng)于一個(gè)謂詞,即斷言流中的元素滿足某個(gè)條件,返回一個(gè) 布爾值
 */
 Stream<T> filter(Predicate<? super T> predicate);

具體使用方法如下:

 public class Main {
 
     public static void main(String[] args) {
         List<String> list = MyUtil.getList();
         System.out.println("過濾操作之前:");
         System.out.println(list);
         // 過濾不以 a 開頭的字符串,collect() 將流中的元素放到一個(gè)新的集合中
         List<String> newList = list.stream().filter(s -> !s.startsWith("a")).collect(Collectors.toList());
         System.out.println("-------------------------");
         System.out.println("過濾操作之后:");
         System.out.println(newList);
     }
 }
 
 ======== 輸出 =========
 過濾操作之前:
 [abc, xyz, fgh, abc, def, xyz, efg]
 -------------------------
 過濾操作之后:
 [xyz, fgh, def, xyz, efg]

Sorted

sorted 可以幫助我們排序流中的元素,方法簽名如下:

 /*
 中間操作,傳入一個(gè) Comparator,對流中的元素進(jìn)行排序,如果不傳入,則使用默認(rèn)的 Comparable 排序
 對原集合不影響
 */
 Stream<T> sorted(Comparator<? super T> comparator);

具體使用方法如下:

 public class Main {
 
     public static void main(String[] args) {
         List<String> list = MyUtil.getList();
         System.out.println("排序操作之前:");
         System.out.println(list);
         List<String> newList = list.stream().sorted().collect(Collectors.toList());
         System.out.println("-------------------------");
         System.out.println("排序操作之后:");
         System.out.println(newList);
         System.out.println("自定義排序:");
         // 倒序排序。 forEach 方法可以用傳入的方法 逐個(gè) 處理流中的元素
         list.stream().sorted((s1, s2)-> -s1.compareTo(s2)).forEach(System.out::println);
     }
 }
 
 ======== 輸出 =========
 排序操作之前:
 [abc, xyz, fgh, abc, def, xyz, efg]
 -------------------------
 排序操作之后:
 [abc, abc, def, efg, fgh, xyz, xyz]
 自定義排序:
 xyz
 xyz
 fgh
 efg
 def
 abc
 abc

Map

Map 操作可以幫助我們將流中的一類元素映射為另一類元素,最典型的應(yīng)用就是可以用來將數(shù)據(jù)庫實(shí)體類轉(zhuǎn)換為供前端使用的 DTO 類。方法簽名如下:

 /*
 中間操作,可以將一個(gè)對象轉(zhuǎn)化為另一個(gè)對象
 例如做 DTO 數(shù)據(jù)轉(zhuǎn)換
 */
 <R> Stream<R> map(Function<? super T, ? extends R> mapper);

具體使用方法如下:

 public class Main {
 
     public static void main(String[] args) {
         List<Student> students = MyUtil.getStudents();
         System.out.println("map 操作之前");
         System.out.println(students);
         // collect 方法可以將流中的元素收集到一個(gè) Collection 中,如果有去除重復(fù)元素的需求,可以考慮收集到 Set 中
         List<StudentDTO> dtos = students.stream().map(student -> {
             StudentDTO dto = new StudentDTO();
             dto.setName(student.getName());
             dto.setMajor(student.getMajor());
             return dto;
         }).collect(Collectors.toList());
         System.out.println("-------------------------");
         System.out.println("map 操作之后");
         System.out.println(dtos);
     }
 }
 
 ======== 輸出 =========
 map 操作之前
 [Student{id='cb5726cd-e73a-443e-95e5-155aa6e876ae', name='張三', age='16', major='計(jì)算機(jī)科學(xué)與技術(shù)'}, Student{id='94478bae-b2ee-4c43-bac0-12f45f4099cd', name='李四', age='18', major='物聯(lián)網(wǎng)工程'}, Student{id='5fdd9e19-f7cf-4c61-b506-0ef58a36dcbe', name='王五', age='20', major='網(wǎng)絡(luò)工程'}]
 -------------------------
 map 操作之后
 [StudentDTO{name='張三', major='計(jì)算機(jī)科學(xué)與技術(shù)'}, StudentDTO{name='李四', major='物聯(lián)網(wǎng)工程'}, StudentDTO{name='王五', major='網(wǎng)絡(luò)工程'}]

Match

 /*
 終端操作,可以用來匹配操作,返回一個(gè) boolean 值
 可以方便地匹配集合中是否存在某種元素
 */
 // 只要集合中有一個(gè)匹配,就返回 true
 boolean anyMatch(Predicate<? super T> predicate);
 // 集合中所有元素都匹配,才返回 true
 boolean allMatch(Predicate<? super T> predicate);
 // 集合中所有元素都不匹配,返回 true
 boolean noneMatch(Predicate<? super T> predicate);

具體使用方法如下:

 public class Main {
 
     public static void main(String[] args) {
         List<String> list = MyUtil.getList();
         System.out.println("集合中的所有元素是否都以 a 開頭");
         System.out.println(list.stream().allMatch(s -> s.startsWith("a")));
 
         System.out.println("集合中是否存在元素以 a 開頭");
         System.out.println(list.stream().anyMatch(s -> s.startsWith("a")));
 
         System.out.println("集合中的元素是否都不以 a 開頭(相當(dāng)于 allMatch 的取反):");
         System.out.println(list.stream().noneMatch(s -> s.startsWith("a")));
     }
 }
 
 ======== 輸出 =========
 集合中的所有元素是否都以 a 開頭
 false
 集合中是否存在元素以 a 開頭
 true
 集合中的元素是否都不以 a 開頭(相當(dāng)于 allMatch 的取反):
 false

Count

 /*
 終端操作,返回 stream 流中及集合中的元素個(gè)數(shù),返回一個(gè) long 類型
 */
 long count();

具體使用方法如下:

 public class Main {
 
     public static void main(String[] args) {
         List<String> list = MyUtil.getList();
         System.out.println(list);
         System.out.println("集合中的個(gè)數(shù):" + list.size());
 
         long count = list.stream().filter(s -> s.startsWith("a")).count();
         System.out.println("集合中以 a 開頭的元素個(gè)數(shù):" + count);
     }
 }
 
 ======== 輸出 =========
 [abc, xyz, fgh, abc, def, xyz, efg]
 集合中的個(gè)數(shù):7
 集合中以 a 開頭的元素個(gè)數(shù):2

Reduce

 /*
 終端操作,可以理解為減少集合的個(gè)數(shù),對集合中的元素不斷進(jìn)行累加,最終只得到一個(gè)元素
 Optional 包含一個(gè)對象,可以防止空指針異常
 */
 Optional<T> reduce(BinaryOperator<T> accumulator);

具體使用方法如下:

 public class Main {
 
     public static void main(String[] args) {
         List<String> list = MyUtil.getList();
         // 可以理解為減少集合的個(gè)數(shù),對集合中的元素不斷進(jìn)行累加,最終只得到一個(gè)元素
         // 例如對數(shù)字集合進(jìn)行累加進(jìn)行求和
         String s = list.stream().reduce((s1, s2) -> s1 + "###" + s2).get();
         System.out.println(s);
     }
 }
 
 ======== 輸出 =========
 abc###xyz###fgh###abc###def###xyz###efg

關(guān)于“Java集合Stream流如何使用”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識(shí),可以關(guān)注億速云行業(yè)資訊頻道,小編每天都會(huì)為大家更新不同的知識(shí)點(diǎn)。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI