JDK1.8新特性Stream流式操作的具體使用
一. 前言
隨著Java的發(fā)展,越來(lái)越多的企業(yè)開始使用JDK1.8 版本。JDK1.8 是自 JDK1.5之后最重要的版本,這個(gè)版本包含語(yǔ)言、編譯器、庫(kù)、工具、JVM等方面的十多個(gè)新特性。本次文章將著重學(xué)習(xí)Stream。
Stream 是JDK1.8 中處理集合的關(guān)鍵抽象概念,Lambda 和 Stream 是JDK1.8新增的函數(shù)式編程最有亮點(diǎn)的特性了,它可以指定你希望對(duì)集合進(jìn)行的操作,可以執(zhí)行非常復(fù)雜的查找、過(guò)濾和映射數(shù)據(jù)等操作。使用Stream API 對(duì)集合數(shù)據(jù)進(jìn)行操作,就類似于使用SQL執(zhí)行的數(shù)據(jù)庫(kù)查詢。Stream 使用一種類似用 SQL 語(yǔ)句從數(shù)據(jù)庫(kù)查詢數(shù)據(jù)的直觀方式來(lái)提供一種對(duì) Java 集合運(yùn)算和表達(dá)的高階抽象。Stream API可以極大提高Java程序員的生產(chǎn)力,讓程序員寫出高效率、干凈、簡(jiǎn)潔的代碼。
這種風(fēng)格將要處理的元素集合看作一種流, 流在管道中傳輸, 并且可以在管道的節(jié)點(diǎn)上進(jìn)行處理, 比如篩選, 排序,聚合等。
元素流在管道中經(jīng)過(guò)中間操作(intermediate operation)的處理,最后由最終操作(terminal operation)得到前面處理的結(jié)果。
+--------------------+ +------+ +------+ +---+ +-------+ | stream of elements +-----> |filter+-> |sorted+-> |map+-> |collect| +--------------------+ +------+ +------+ +---+ +-------+
簡(jiǎn)而言之,Stream API提供了一種高效且易于使用的處理數(shù)據(jù)的方式。
二. 什么是Stream
1.Stream(流)是一個(gè)來(lái)自數(shù)據(jù)源的元素隊(duì)列并支持聚合操作。
- 元素是特定類型的對(duì)象,形成一個(gè)隊(duì)列。 Java中的Stream并不會(huì)存儲(chǔ)元素,而是按需計(jì)算。
- 數(shù)據(jù)源 流的來(lái)源。 可以是集合,數(shù)組,I/O channel, 產(chǎn)生器generator 等。
- 聚合操作 類似SQL語(yǔ)句一樣的操作, 比如filter, map, reduce, find, match, sorted等。
2.和以前的Collection操作不同, Stream操作還有兩個(gè)基礎(chǔ)的特征:
- Pipelining: 中間操作都會(huì)返回流對(duì)象本身。 這樣多個(gè)操作可以串聯(lián)成一個(gè)管道, 如同流式風(fēng)格(fluent style)。 這樣做可以對(duì)操作進(jìn)行優(yōu)化, 比如延遲執(zhí)行(laziness)和短路( short-circuiting)。
- 內(nèi)部迭代: 以前對(duì)集合遍歷都是通過(guò)Iterator或者For-Each的方式, 顯式的在集合外部進(jìn)行迭代, 這叫做外部迭代。 Stream提供了內(nèi)部迭代的方式, 通過(guò)訪問(wèn)者模式(Visitor)實(shí)現(xiàn)。
特點(diǎn):
1.Stream 不是數(shù)據(jù)結(jié)構(gòu),不會(huì)保存數(shù)據(jù)。
2.Stream 不會(huì)修改原來(lái)的數(shù)據(jù)源,它會(huì)將操作后的數(shù)據(jù)保存到另外一個(gè)對(duì)象中。(保留意見:畢竟peek方法可以修改流中元素)
3.惰性求值,流在中間處理過(guò)程中,只是對(duì)操作進(jìn)行了記錄,并不會(huì)立即執(zhí)行,需要等到執(zhí)行終止操作的時(shí)候才會(huì)進(jìn)行實(shí)際的計(jì)算。
三. 關(guān)于Stream API
1. Stream API分類
Stream 操作分為中間操作或者終止操作兩種,終止操作返回一特定類型的計(jì)算結(jié)果,而中間操作返回Stream本身,Stream 的操作類型具體分類如下:
解釋:
- 無(wú)狀態(tài):指元素的處理不受之前元素的影響;
- 有狀態(tài):指該操作只有拿到所有元素之后才能繼續(xù)下去。
- 非短路操作:指必須處理所有元素才能得到最終結(jié)果;
- 短路操作:指遇到某些符合條件的元素就可以得到最終結(jié)果,如 A || B,只要A為true,則無(wú)需判斷B的結(jié)果。
2. 如何使用Stream流?
使用Stream流分為三步。
創(chuàng)建Stream
一個(gè)數(shù)據(jù)源(如:集合、數(shù)組),獲取一個(gè)Stream流。
中間操作
一個(gè)中間操作鏈,對(duì)數(shù)據(jù)源(如:集合、數(shù)組)的數(shù)據(jù)進(jìn)行處理。
終止操作
一個(gè)終止操作,執(zhí)行中間操作鏈,并產(chǎn)生一個(gè)計(jì)算結(jié)果。
3. Stream的中間操作和結(jié)束操作
中間操作
filter: 過(guò)濾流,過(guò)濾流中的元素,返回一個(gè)符合條件的Stream
map: 轉(zhuǎn)換流,將一種類型的流轉(zhuǎn)換為另外一種流。(mapToInt、mapToLong、mapToDouble 返回int、long、double基本類型對(duì)應(yīng)的Stream)
flatMap:簡(jiǎn)單的說(shuō),就是一個(gè)或多個(gè)流合并成一個(gè)新流。(flatMapToInt、flatMapToLong、flatMapToDouble 返回對(duì)應(yīng)的IntStream、LongStream、DoubleStream流。)
distinct: 返回去重的Stream。
sorted: 返回一個(gè)排序的Stream。
peek: 主要用來(lái)查看流中元素的數(shù)據(jù)狀態(tài)。
limit: 返回前n個(gè)元素?cái)?shù)據(jù)組成的Stream。屬于短路操作
skip: 返回第n個(gè)元素后面數(shù)據(jù)組成的Stream。
結(jié)束操作
forEach: 循環(huán)操作Stream中數(shù)據(jù)。
toArray: 返回流中元素對(duì)應(yīng)的數(shù)組對(duì)象。
reduce: 聚合操作,用來(lái)做統(tǒng)計(jì)。
collect: 聚合操作,封裝目標(biāo)數(shù)據(jù)。
min、max、count: 聚合操作,最小值,最大值,總數(shù)量。
anyMatch: 短路操作,有一個(gè)符合條件返回true。
allMatch: 所有數(shù)據(jù)都符合條件返回true。
noneMatch: 所有數(shù)據(jù)都不符合條件返回true。
findFirst: 短路操作,獲取第一個(gè)元素。
findAny: 短路操作,獲取任一元素。
forEachOrdered: 暗元素順序執(zhí)行循環(huán)操作。
四. 如何獲取Stream流
在 Java 8 中, 集合接口有兩個(gè)方法來(lái)生成流:
- stream() − 為集合創(chuàng)建串行流。
- parallelStream() − 為集合創(chuàng)建并行流。
1. 常見幾種集合流的創(chuàng)建
/** * stream,獲取各種集合的stream流 */ @Test public void testCollectionStream(){ //List集合 List<String> stringList = new ArrayList<>(); //Set集合 Set<String> stringSet = new HashSet<>(); //Map集合 Map<String,Object> stringObjectMap = new HashMap<>(); //數(shù)組 String[] stringArray = {"張三三","李四","王五","王五","趙八",}; //通過(guò)list獲取stream流 Stream<String> streamList = stringList.stream(); //通過(guò)set獲取stream流 Stream<String> streamSet = stringSet.stream(); //通過(guò)map獲取stream流 Stream<String> streamMap = stringObjectMap.keySet().stream(); //通過(guò)array獲取stream流 Stream<String> streamArray1 = Stream.of(stringArray); }
2. 構(gòu)造流的幾種常見方法
@Test public void testCollectionStream(){ // 1. Individual values Stream stream = Stream.of("a", "b", "c"); // 2. Arrays String[] strArray = new String[]{"a", "b", "c"}; stream = Stream.of(strArray); stream = Arrays.stream(strArray); // 3. Collections List<String> list = Arrays.asList(strArray); stream = list.stream(); }
五. Stream在代碼中的使用方式
關(guān)于Stream的常見操作方式,主要分為兩大類: 中間操作和終止操作 ,接下來(lái)就通過(guò)這兩大分類,講解下具體的語(yǔ)法用法。
1. 流的中間操作
1.1 篩選過(guò)濾
filter:過(guò)濾流中的某些元素
/** * filter 方法 , 返回符合過(guò)濾條件的值 */ @Test public void testFilter() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.stream().filter(e -> e.contains("張")).forEach(System.out::println); }
或
filter多個(gè)過(guò)濾篩選條件
/** * list集合stream流式操作 */ @Test public void testStreamList() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.stream().filter(e -> e.startsWith("張")) //過(guò)濾所有姓張的人 .filter(e -> e.length() == 3) //過(guò)濾所有姓名是3個(gè)字的人 .forEach(System.out::println); //遍歷打印,System.out::println表明System.out調(diào)用println打印方法 }
limit(n):獲取前n個(gè)元素
/** * limit 方法 ,返回前n個(gè)元素?cái)?shù)據(jù)值組成的Stream。 */ @Test public void testLimit() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.add("王二麻子"); list.stream().limit(3).forEach(System.out::println); //取前3個(gè) }
skip(n):跳過(guò)n元素,配合limit(n)可實(shí)現(xiàn)分頁(yè)
/** * skip方法 ,跳過(guò)前n個(gè)元素的中間流操作,返回剩下的值。 */ @Test public void testSkip() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.add("王二麻子"); //list.stream().skip(3).forEach(System.out::println); //跳過(guò)前3個(gè) list.stream().skip(3).limit(2).forEach(System.out::println); //skip+limit實(shí)現(xiàn)分頁(yè) }
distinct:通過(guò)流中元素的 hashCode() 和 equals() 去除重復(fù)元素
/** * distinct, 返回去重的Stream */ @Test public void testDistinct() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("李四"); list.add("王五"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.add("王二麻子"); list.stream().distinct().collect(Collectors.toList()).forEach(System.out::println); }
1.2 排序
sorted():自然排序,流中元素需實(shí)現(xiàn)Comparable接口
/** * sorted: 返回一個(gè)排序的Stream */ @Test public void testSorted() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("李四"); list.add("王五"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.add("王二麻子"); list.stream().distinct().sorted().collect(Collectors.toList()).forEach(System.out::println); }
sorted(Comparator com):定制排序,自定義Comparator排序器
1.3 映射
map:接收一個(gè)函數(shù)作為參數(shù),該函數(shù)會(huì)被應(yīng)用到每個(gè)元素上,并將其映射成一個(gè)新的元素。
/** * 遍歷map集合,截取substring(2)開始的值 */ @Test public void testMap() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.add("王二麻子"); Stream<String> stream = list.stream().map(e -> e.substring(2)); stream.forEach(System.out::println); }
forEach:ForEach流式遍歷集合
/** * forEach, ForEach流式遍歷list集合 */ @Test public void testForEach() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.stream().forEach(System.out::println); }
2. 流的終止操作
2.1 匹配、聚合操作
allMatch:接收一個(gè) Predicate 函數(shù),當(dāng)流中每個(gè)元素都符合該斷言時(shí)才返回true,否則返回false
/** * allMatch:接收一個(gè) Predicate 函數(shù),當(dāng)流中每個(gè)元素都符合該斷言時(shí)才返回true,否則返回false */ @Test public void testAllMatch() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("李四"); list.add("王五"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.add("王二麻子"); boolean b = list.stream() .allMatch(e -> list.size() > 8); System.out.println("b = " + b); }
noneMatch:接收一個(gè) Predicate 函數(shù),當(dāng)流中每個(gè)元素都不符合該斷言時(shí)才返回true,否則返回false
/** * noneMatch: 接收一個(gè) Predicate 函數(shù),當(dāng)流中每個(gè)元素都不符合該斷言時(shí)才返回true,否則返回false */ @Test public void testNoneMatch() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("李四"); list.add("王五"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.add("王二麻子"); boolean b = list.stream().noneMatch(e->e.equals("張三")); System.out.println("b = " + b); }
anyMatch:接收一個(gè) Predicate 函數(shù),只要流中有一個(gè)元素滿足該斷言則返回true,否則返回false
/** * anyMatch:接收一個(gè) Predicate 函數(shù),只要流中有一個(gè)元素滿足該斷言則返回true,否則返回false */ @Test public void testAnyMatch() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("李四"); list.add("王五"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.add("王二麻子"); boolean b = list.stream().anyMatch(e -> e.equals("王二麻子")); System.out.println("b = " + b); }
findFirst:返回流中第一個(gè)元素
/** * findFirst:返回流中第一個(gè)元素 */ @Test public void testFindFirsth() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("李四"); list.add("王五"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.add("王二麻子"); Optional<String> first = list.stream().findFirst(); System.out.println("first = " + first.get()); }
findAny:返回流中的任意元素
/** * findAny:返回流中第一個(gè)元素 */ @Test public void testFindAny() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("李四"); list.add("王五"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.add("王二麻子"); Optional<String> any = list.stream().findAny(); System.out.println("any = " + any.get()); }
count:返回流中元素的總個(gè)數(shù)
/** * count,獲取List集合的長(zhǎng)度 */ @Test public void testCount() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.add("王二麻子"); long count = list.stream().count(); System.out.println("count = " + count); int size = list.size(); System.out.println("size = " + size); }
2.2 Collector 工具庫(kù):Collectors
Collectors
Student s1 = new Student("aa", 10,1); Student s2 = new Student("bb", 20,2); Student s3 = new Student("cc", 10,3); List<Student> list = Arrays.asList(s1, s2, s3); //裝成list List<Integer> ageList = list.stream().map(Student::getAge).collect(Collectors.toList()); // [10, 20, 10] //轉(zhuǎn)成set Set<Integer> ageSet = list.stream().map(Student::getAge).collect(Collectors.toSet()); // [20, 10] //轉(zhuǎn)成map,注:key不能相同,否則報(bào)錯(cuò) Map<String, Integer> studentMap = list.stream().collect(Collectors.toMap(Student::getName, Student::getAge)); // {cc=10, bb=20, aa=10} //字符串分隔符連接 String joinName = list.stream().map(Student::getName).collect(Collectors.joining(",", "(", ")")); // (aa,bb,cc) //聚合操作 //1.學(xué)生總數(shù) Long count = list.stream().collect(Collectors.counting()); // 3 //2.最大年齡 (最小的minBy同理) Integer maxAge = list.stream().map(Student::getAge).collect(Collectors.maxBy(Integer::compare)).get(); // 20 //3.所有人的年齡 Integer sumAge = list.stream().collect(Collectors.summingInt(Student::getAge)); // 40 //4.平均年齡 Double averageAge = list.stream().collect(Collectors.averagingDouble(Student::getAge)); // 13.333333333333334 // 帶上以上所有方法 DoubleSummaryStatistics statistics = list.stream().collect(Collectors.summarizingDouble(Student::getAge)); System.out.println("count:" + statistics.getCount() + ",max:" + statistics.getMax() + ",sum:" + statistics.getSum() + ",average:" + statistics.getAverage()); //分組 Map<Integer, List<Student>> ageMap = list.stream().collect(Collectors.groupingBy(Student::getAge)); //多重分組,先根據(jù)類型分再根據(jù)年齡分 Map<Integer, Map<Integer, List<Student>>> typeAgeMap = list.stream().collect(Collectors.groupingBy(Student::getType, Collectors.groupingBy(Student::getAge))); //分區(qū) //分成兩部分,一部分大于10歲,一部分小于等于10歲 Map<Boolean, List<Student>> partMap = list.stream().collect(Collectors.partitioningBy(v -> v.getAge() > 10)); //規(guī)約 Integer allAge = list.stream().map(Student::getAge).collect(Collectors.reducing(Integer::sum)).get(); //40
六、Stream操作代碼
為了方便小伙伴們看到這篇博客時(shí),學(xué)習(xí)的更加輕松,這里貼出源碼,小伙伴們學(xué)習(xí)是可貼到IDEA運(yùn)行查看Stream過(guò)濾篩選的結(jié)果,以此對(duì)Stream的流式操作更加熟悉。
package com.java8.example.chapter3; import org.junit.jupiter.api.Test; import java.util.*; import java.util.stream.Collectors; import java.util.stream.Stream; /** * @desc: Stream流式操作 * @author: cao_wencao * @date: 2020-09-17 15:24 */ public class TestStreamList { /** * list集合stream流式操作 */ @Test public void testStreamList() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.stream().filter(e -> e.startsWith("張")) //過(guò)濾所有姓張的人 .filter(e -> e.length() == 3) //過(guò)濾所有姓名是3個(gè)字的人 .forEach(System.out::println); //遍歷打印,System.out::println表明System.out調(diào)用println打印方法 } /** * stream,獲取各種集合的stream流 */ @Test public void testCollectionStream() { List<String> stringList = new ArrayList<>(); Set<String> stringSet = new HashSet<>(); Map<String, Object> stringObjectMap = new HashMap<>(); String[] stringArray = {"張三三", "李四", "王五", "王五", "趙八",}; //通過(guò)list獲取stream流 Stream<String> streamList = stringList.stream(); //通過(guò)set獲取stream流 Stream<String> streamSet = stringSet.stream(); //通過(guò)map獲取stream流 Stream<String> streamMap = stringObjectMap.keySet().stream(); //通過(guò)array獲取stream流 Stream<String> streamArray1 = Stream.of(stringArray); } /** * forEach, ForEach流式遍歷list集合 */ @Test public void testForEach() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.stream().forEach(System.out::println); } /** * filter 方法 , 返回符合過(guò)濾條件的值 */ @Test public void testFilter() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.stream().filter(e -> e.contains("張")).forEach(System.out::println); } /** * 遍歷map集合,截取substring(2)開始的值 */ @Test public void testMap() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.add("王二麻子"); Stream<String> stream = list.stream().map(e -> e.substring(2)); stream.forEach(System.out::println); } /** * count,獲取List集合的長(zhǎng)度 */ @Test public void testCount() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.add("王二麻子"); long count = list.stream().count(); System.out.println("count = " + count); int size = list.size(); System.out.println("size = " + size); } /** * limit 方法 ,返回前n個(gè)元素?cái)?shù)據(jù)值組成的Stream。 */ @Test public void testLimit() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.add("王二麻子"); list.stream().limit(3).forEach(System.out::println); //取前3個(gè) } /** * skip方法 ,跳過(guò)前n個(gè)元素的中間流操作,返回剩下的值。 */ @Test public void testSkip() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.add("王二麻子"); //list.stream().skip(3).forEach(System.out::println); //跳過(guò)前3個(gè) list.stream().skip(3).limit(2).forEach(System.out::println); //skip+limit實(shí)現(xiàn)分頁(yè) } /** * collect,將流轉(zhuǎn)化為集合 */ @Test public void testCollect() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.add("王二麻子"); List<String> collect = list.stream().skip(3).limit(2).collect(Collectors.toList()); collect.forEach(System.out::println); } /** * distinct, 返回去重的Stream */ @Test public void testDistinct() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("李四"); list.add("王五"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.add("王二麻子"); list.stream().distinct().collect(Collectors.toList()).forEach(System.out::println); } /** * sorted: 返回一個(gè)排序的Stream */ @Test public void testSorted() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("李四"); list.add("王五"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.add("王二麻子"); list.stream().distinct().sorted().collect(Collectors.toList()).forEach(System.out::println); } /** * anyMatch:接收一個(gè) Predicate 函數(shù),只要流中有一個(gè)元素滿足該斷言則返回true,否則返回false */ @Test public void testAnyMatch() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("李四"); list.add("王五"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.add("王二麻子"); boolean b = list.stream().anyMatch(e -> e.equals("王二麻子")); System.out.println("b = " + b); } /** * noneMatch: 接收一個(gè) Predicate 函數(shù),當(dāng)流中每個(gè)元素都不符合該斷言時(shí)才返回true,否則返回false */ @Test public void testNoneMatch() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("李四"); list.add("王五"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.add("王二麻子"); boolean b = list.stream().noneMatch(e->e.equals("張三")); System.out.println("b = " + b); } /** * allMatch:接收一個(gè) Predicate 函數(shù),當(dāng)流中每個(gè)元素都符合該斷言時(shí)才返回true,否則返回false */ @Test public void testAllMatch() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("李四"); list.add("王五"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.add("王二麻子"); boolean b = list.stream() .allMatch(e -> list.size() > 8); System.out.println("b = " + b); } /** * findFirst:返回流中第一個(gè)元素 */ @Test public void testFindFirsth() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("李四"); list.add("王五"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.add("王二麻子"); Optional<String> first = list.stream().findFirst(); System.out.println("first = " + first.get()); } /** * findAny:返回流中第一個(gè)元素 */ @Test public void testFindAny() { List<String> list = new ArrayList<>(); list.add("張三三"); list.add("李四"); list.add("李四"); list.add("王五"); list.add("王五"); list.add("孫七"); list.add("趙八"); list.add("王二麻子"); Optional<String> any = list.stream().findAny(); System.out.println("any = " + any.get()); } /** * max:返回流中元素最大值 */ @Test public void testMax() { List<Integer> list = new ArrayList<>(); list.add(11); list.add(22); list.add(33); list.add(44); list.add(55); list.add(66); list.add(77); list.add(88); Integer integer = list.stream().max(Integer::compareTo).get(); System.out.println("integer = " + integer); } /** * min:返回流中元素最小值 */ @Test public void testMin() { List<Integer> list = new ArrayList<>(); list.add(11); list.add(22); list.add(33); list.add(44); list.add(55); list.add(66); list.add(77); list.add(88); Integer integer = list.stream().min(Integer::compareTo).get(); System.out.println("integer = " + integer); list.stream().limit(1).limit(2).distinct().skip(3).filter(f -> f.equals(55)).forEach(System.out::println); } }
總結(jié)
以上就是對(duì)于JDK1.8中Stream流式計(jì)算的一個(gè)簡(jiǎn)單介紹,關(guān)于JDK1.8中的新特性,用法遠(yuǎn)遠(yuǎn)不止這些,這篇文章只作為一個(gè)簡(jiǎn)單的入門使用,更多相關(guān)Java8 Stream流式操作內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
IDEA錯(cuò)誤:找不到或無(wú)法加載主類的完美解決方法
使用IDEA開始就一直在搭建java環(huán)境,許久沒(méi)有使用過(guò)java,剛開始有些生疏,先建了一個(gè)最簡(jiǎn)單的類可是運(yùn)行的時(shí)候出現(xiàn)錯(cuò)誤:找不到或無(wú)法加載主類,下面這篇文章主要給大家介紹了關(guān)于IDEA錯(cuò)誤:找不到或無(wú)法加載主類的完美解決方法,需要的朋友可以參考下2022-07-07Java實(shí)現(xiàn)簡(jiǎn)單的掃雷小程序
這篇文章主要為大家詳細(xì)介紹了Java實(shí)現(xiàn)簡(jiǎn)單的掃雷小程序,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-04-04基于java中byte數(shù)組與int類型的轉(zhuǎn)換(兩種方法)
下面小編就為大家?guī)?lái)一篇基于java中byte數(shù)組與int類型的轉(zhuǎn)換(兩種方法)。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-08-08Spring代理對(duì)象導(dǎo)致的獲取不到原生對(duì)象注解的解決
本文主要介紹了Spring代理對(duì)象導(dǎo)致的獲取不到原生對(duì)象注解的解決,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-04-04spring aop action中驗(yàn)證用戶登錄狀態(tài)的實(shí)例代碼
本篇文章主要介紹了spring aop action中驗(yàn)證用戶登錄狀態(tài)的實(shí)例代碼,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-07-07