成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

Java 8 Stream實(shí)踐

JerryWangSAP / 1721人閱讀

摘要:前面的話中的于版本析出,平時(shí)項(xiàng)目中也有用到,今天就系統(tǒng)的來(lái)實(shí)踐一下??梢愿鶕?jù)隊(duì)員的年齡進(jìn)行分組執(zhí)行結(jié)果為費(fèi)爾南多費(fèi)爾南迪尼奧隋維杰克魯伊夫卡爾德克阿德里安結(jié)果是一個(gè),為不重復(fù)的年齡,為屬于該年齡的隊(duì)員列表。已經(jīng)實(shí)現(xiàn)了分組。

前面的話】Java中的Stream于1.8版本析出,平時(shí)項(xiàng)目中也有用到,今天就系統(tǒng)的來(lái)實(shí)踐一下。下面借用重慶力帆隊(duì)伍中我個(gè)人比較喜歡的球員來(lái)操作一波,隊(duì)員的年齡為了便于展示某些api做了調(diào)整,請(qǐng)不要太認(rèn)真哦。

壹. Stream理解

在java中我們稱Stream為『』,我們經(jīng)常會(huì)用流去對(duì)集合進(jìn)行一些流水線的操作。stream就像工廠一樣,只需要把集合、命令還有一些參數(shù)灌輸?shù)搅魉€中去,就可以加工成得出想要的結(jié)果。這樣的流水線能大大簡(jiǎn)潔代碼,減少操作。給我個(gè)人的感覺(jué)類似JavaScript中的鏈?zhǔn)胶瘮?shù)。

貳. Stream流程
原集合 —> 流 —> 各種操作(過(guò)濾、分組、統(tǒng)計(jì)) —> 終端操作

Stream流的操作流程一般都是這樣的,先將集合轉(zhuǎn)為流,然后經(jīng)過(guò)各種操作,比如過(guò)濾、篩選、分組、計(jì)算。最后的終端操作,就是轉(zhuǎn)化成我們想要的數(shù)據(jù),這個(gè)數(shù)據(jù)的形式一般還是集合,有時(shí)也會(huì)按照需求輸出count計(jì)數(shù)。下文會(huì)一一舉例。

叁. API實(shí)踐

首先,定義一個(gè)用戶對(duì)象,包含姓名、年齡、id三個(gè)成員變量:

package com.eelve.training.entity;

import lombok.*;

import javax.persistence.*;

/**
 * @ClassName User
 * @Description TDO
 * @Author zhao.zhilue
 * @Date 2019/6/28 15:21
 * @Version 1.0
 **/
@Data
@Entity
@Table(name = "user")
@ToString
@NoArgsConstructor
@AllArgsConstructor
@EqualsAndHashCode(exclude={"id","name"})
public class User implements  Comparable{
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    @Column(name = "id")
    private Integer id;

    /**
     * Link name.
     */
    @Column(name = "name", columnDefinition = "varchar(255) not null")
    private String name;

    @Column(name = "age")
    private Integer age;

    public User(String name, Integer age) {
        this.name = name;
        this.age = age;
    }
    @Override
    public int compareTo(User o) {
        return age.compareTo(o.getAge());
    }
}

然后在數(shù)據(jù)庫(kù)中插入測(cè)試數(shù)據(jù),見(jiàn)下圖:

3.1過(guò)濾 1)filter 過(guò)濾(T-> boolean)

假如我們要實(shí)現(xiàn)過(guò)濾出40歲以下的隊(duì)員,我們可以這樣來(lái)實(shí)現(xiàn):

@Test
    public void testUserStreamFilter(){
        List userList = userMapper.getALL();
        List resultList = userList.stream().filter(user -> user.getAge() <= 40).collect(Collectors.toList());
        for (User user :  resultList){
            System.out.println(user.toString());
        }
    }

filter里面,->箭頭后面跟著的是一個(gè)boolean值,可以寫(xiě)任何的過(guò)濾條件,就相當(dāng)于sql中where后面的東西,換句話說(shuō),能用sql實(shí)現(xiàn)的功能這里都可以實(shí)現(xiàn)
執(zhí)行結(jié)果為:

User(id=1, name=費(fèi)爾南多, age=25)
User(id=2, name=費(fèi)爾南迪尼奧, age=26)
User(id=3, name=卡爾德克, age=27)
User(id=4, name=阿德里安, age=28)
User(id=5, name=隋維杰, age=26)
2)distinct 去重

其用法和sql中的使用類似,假如我們要實(shí)現(xiàn)過(guò)去除用重復(fù)年齡的隊(duì)員,我們可以這樣來(lái)實(shí)現(xiàn):

@Test
    public void testUserDistinct(){
        List userList = userMapper.getALL();
        List resultList = userList.stream().distinct().collect(Collectors.toList());
        for (User user :  resultList){
            System.out.println(user.toString());
        }
    }

執(zhí)行結(jié)果為:

User(id=1, name=費(fèi)爾南多, age=25)
User(id=2, name=費(fèi)爾南迪尼奧, age=26)
User(id=3, name=卡爾德克, age=27)
User(id=4, name=阿德里安, age=28)
User(id=6, name=克魯伊夫, age=43)
3)sorted排序

如果流中的元素的類實(shí)現(xiàn)了 Comparable 接口,即有自己的排序規(guī)則,那么可以直接調(diào)用 sorted() 方法對(duì)元素進(jìn)行排序,如:

 @Override
    public int compareTo(User o) {
        return age.compareTo(o.getAge());
    }
@Test
    public void testUserStreamSorted(){
        List userList = userMapper.getALL();
        List resultList = userList.stream().sorted().collect(Collectors.toList());
        for (User user :  resultList){
            System.out.println(user.toString());
        }
    }

反之, 需要調(diào)用 sorted((T, T) -> int) 實(shí)現(xiàn) Comparator 接口。

@Test
    public void testUserStreamSortedWithComparator(){
        List userList = userMapper.getALL();
        List resultList = userList.stream().sorted(Comparator.comparingInt(User::getAge)).collect(Collectors.toList());
        for (User user :  resultList){
            System.out.println(user.toString());
        }
    }

執(zhí)行結(jié)果為:

User(id=1, name=費(fèi)爾南多, age=25)
User(id=2, name=費(fèi)爾南迪尼奧, age=26)
User(id=5, name=隋維杰, age=26)
User(id=3, name=卡爾德克, age=27)
User(id=4, name=阿德里安, age=28)
User(id=6, name=克魯伊夫, age=43)
4)limit() 返回前n個(gè)元素

如果想知道隊(duì)伍中年齡最小的就可以使用下面來(lái)實(shí)現(xiàn):

@Test
    public void testUserStreamLimit(){
        List userList = userMapper.getALL();
        List resultList = userList.stream().limit(2).collect(Collectors.toList());
        for (User user :  resultList){
            System.out.println(user.toString());
        }
    }

執(zhí)行結(jié)果為:

User(id=1, name=費(fèi)爾南多, age=25)
User(id=2, name=費(fèi)爾南迪尼奧, age=26)
5)skip

它的用法和limit正好相反,是去除前面幾個(gè)元素。
假如我們要去除前面兩個(gè)元素就可以使用下面的方法來(lái)實(shí)現(xiàn):

@Test
    public void testUserStreamSkip(){
        List userList = userMapper.getALL();
        List resultList = userList.stream().skip(2).collect(Collectors.toList());
        for (User user :  resultList){
            System.out.println(user.toString());
        }
    }

執(zhí)行結(jié)果為:

User(id=3, name=卡爾德克, age=27)
User(id=4, name=阿德里安, age=28)
User(id=5, name=隋維杰, age=26)
User(id=6, name=克魯伊夫, age=43)
6)組合使用

以上的過(guò)濾函數(shù)物品們可以組合來(lái)使用來(lái)實(shí)現(xiàn)我們具體的需求,示例代碼如下:

 @Test
    public void testUserStreamSortLimit(){
        List userList = userMapper.getALL();
        List resultList = userList.stream().sorted().limit(5).collect(Collectors.toList());
        for (User user :  resultList){
            System.out.println(user.toString());
        }
    }

這樣我們就可以得到先排序后限制的結(jié)果:

User(id=1, name=費(fèi)爾南多, age=25)
User(id=2, name=費(fèi)爾南迪尼奧, age=26)
User(id=5, name=隋維杰, age=26)
User(id=3, name=卡爾德克, age=27)
User(id=4, name=阿德里安, age=28)
3.2 映射 1)map(T->R)

map是將T類型的數(shù)據(jù)轉(zhuǎn)為R類型的數(shù)據(jù),比如我們想要設(shè)置一個(gè)新的list,存儲(chǔ)用戶所有的城市信息。

@Test
    public void testUserStreamMap(){
        List userList = userMapper.getALL();
        List resultList = userList.stream().map(User::getAge).distinct().collect(Collectors.toList());
        System.out.println(resultList.toString());
    }

這樣我們可以得到所有年齡的樣本,執(zhí)行結(jié)果為:

[25, 26, 27, 28, 43]
2)flatMap(T -> Stream)

將流中的每一個(gè)元素 T 映射為一個(gè)流,再把每一個(gè)流連接成為一個(gè)流。

@Test
    public void testStreamMap(){
        List habitsList = new ArrayList<>();
        habitsList.add("唱歌,聽(tīng)歌");
        habitsList.add("羽毛球,足球,登山");
        habitsList = habitsList.stream().map(s -> s.split(",")).flatMap(Arrays::stream).collect(Collectors.toList());
        System.out.println(habitsList);
    }

執(zhí)行結(jié)果為:

[唱歌, 聽(tīng)歌, 羽毛球, 足球, 登山]

這里原集合中的數(shù)據(jù)由逗號(hào)分割,使用split進(jìn)行拆分后,得到的是Stream,字符串?dāng)?shù)組組成的流,要使用flatMap的Arrays::stream,將Stream轉(zhuǎn)為Stream,然后把流相連接,組成了完整的唱歌, 聽(tīng)歌, 羽毛球, 足球, 登山。

3.3 查找 1)allMatch(T->boolean)

檢測(cè)是否全部滿足參數(shù)行為,假如我們要檢測(cè)是不是所有隊(duì)員都是U21的球員:

@Test
    public void testUserStreamAllMatch(){
        List userList = userMapper.getALL();
        boolean isNotU21 = userList.stream().allMatch(user -> user.getAge() >= 21);
        System.out.println("是否都不是U21球員:" + isNotU21);
    }

執(zhí)行結(jié)果為:

是否都不是U21球員:true
2)anyMatch(T->boolean)

檢測(cè)是否有任意元素滿足給定的條件,比如,想知道是否有26歲的球員:

@Test
    public void testUserStreamAnyMatch(){
        List userList = userMapper.getALL();
        boolean isAgeU26 = userList.stream().anyMatch(user -> user.getAge() == 26);
        System.out.println("是否有26歲的球員:" + isAgeU26);
    }

執(zhí)行結(jié)果為:

是否有26歲的球員:true
3)noneMatch(T -> boolean)

流中是否有元素匹配給定的 T -> boolean 條件。比如我們要檢測(cè)是否含有U18的隊(duì)員:

 @Test
    public void testUserStreamNoneMatch(){
        List userList = userMapper.getALL();
        boolean isNotU18 = userList.stream().noneMatch(user -> user.getAge() <= 18);
        System.out.println("是否都不是U18球員:" + isNotU18);
    }

執(zhí)行結(jié)果為:

是否都不是U18球員:true

說(shuō)明沒(méi)有U18的隊(duì)員。

4)findFirst( ):找到第一個(gè)元素
@Test
    public void testUserFindFirst(){
        List userList = userMapper.getALL();
        Optional firstUser = userList.stream().sorted().findFirst();
        System.out.println(firstUser.toString());
    }

執(zhí)行結(jié)果為:

Optional[User(id=1, name=費(fèi)爾南多, age=25)]
5)findAny():找到任意一個(gè)元素
@Test
    public void testUserFindAny(){
        List userList = userMapper.getALL();
        Optional anytUser = userList.parallelStream().sorted().findAny();
        System.out.println(anytUser.toString());
    }

執(zhí)行結(jié)果為:

Optional[User(id=2, name=費(fèi)爾南迪尼奧, age=26)]
3.4 歸納計(jì)算 1)求隊(duì)員的總?cè)藬?shù)
@Test
    public void testUserCount(){
        List userList = userMapper.getALL();
        long totalAge = userList.stream().collect(Collectors.counting());
        System.out.println("隊(duì)員人數(shù)為:" + totalAge);
    }

執(zhí)行結(jié)果為:

隊(duì)員人數(shù)為:6
2)得到某一屬性的最大最小值
@Test
    public void testUserMaxAndMin(){
        List userList = userMapper.getALL();
        Optional userMaxAge = userList.stream().collect(Collectors.maxBy(Comparator.comparing(User::getAge)));
        System.out.println("年齡最大的隊(duì)員為:" + userMaxAge.toString());

        Optional userMinAge = userList.stream().collect(Collectors.minBy(Comparator.comparing(User::getAge)));
        System.out.println("年齡最小的隊(duì)員為:" + userMinAge.toString());
    }

執(zhí)行結(jié)果為:

年齡最大的隊(duì)員為:Optional[User(id=6, name=克魯伊夫, age=43)]
年齡最小的隊(duì)員為:Optional[User(id=1, name=費(fèi)爾南多, age=25)]
3)求年齡總和是多少
@Test
    public void testUserSummingInt(){
        List userList = userMapper.getALL();
        int totalAge = userList.stream().collect(Collectors.summingInt(User::getAge));
        System.out.println("年齡總和為:" + totalAge);
    }

執(zhí)行結(jié)果為:

年齡總和為:175

我們經(jīng)常會(huì)用BigDecimal來(lái)記錄金錢(qián),假設(shè)想得到BigDecimal的總和:
// 獲得列表對(duì)象金額, 使用reduce聚合函數(shù),實(shí)現(xiàn)累加器
BigDecimal sum = myList.stream() .map(User::getMoney)
.reduce(BigDecimal.ZERO,BigDecimal::add);

4)求年齡平均值
@Test
    public void testUserAveragingInt(){
        List userList = userMapper.getALL();
        Double totalAge = userList.stream().collect(Collectors.averagingInt(User::getAge));
        System.out.println("平均年齡為:" + totalAge);
    }

執(zhí)行結(jié)果為:

平均年齡為:29.166666666666668
5)一次性得到元素的個(gè)數(shù)、總和、最大值、最小值
@Test
    public void testUserSummarizingInt(){
        List userList = userMapper.getALL();
        IntSummaryStatistics  statistics  = userList.stream().collect(Collectors.summarizingInt(User::getAge));
        System.out.println("年齡的統(tǒng)計(jì)結(jié)果為:" + statistics );
    }

執(zhí)行結(jié)果為:

年齡的統(tǒng)計(jì)結(jié)果為:IntSummaryStatistics{count=6, sum=175, min=25, average=29.166667, max=43}
6)字符串拼接

要將隊(duì)員的姓名連成一個(gè)字符串并用逗號(hào)分割。

@Test
    public void testUserJoining(){
        List userList = userMapper.getALL();
        String  name  = userList.stream().map(User::getName).collect(Collectors.joining(","));
        System.out.println("所有的隊(duì)員名字:" + name );
    }

執(zhí)行結(jié)果為:

所有的隊(duì)員名字:費(fèi)爾南多,費(fèi)爾南迪尼奧,卡爾德克,阿德里安,隋維杰,克魯伊夫
3.5 分組

在數(shù)據(jù)庫(kù)操作中,我們經(jīng)常通過(guò)GROUP BY關(guān)鍵字對(duì)查詢到的數(shù)據(jù)進(jìn)行分組,java8的流式處理也提供了分組的功能。使用Collectors.groupingBy來(lái)進(jìn)行分組。

1)可以根據(jù)隊(duì)員的年齡進(jìn)行分組
 @Test
    public void testUserGroupingBy(){
        List userList = userMapper.getALL();
        Map> ageMap  = userList.stream().collect(Collectors.groupingBy(User::getAge));
        for (Map.Entry> entry :ageMap.entrySet()){
            System.out.println("key= " + entry.getKey() + " and value= " + entry.getValue());
        }
    }

執(zhí)行結(jié)果為:

key= 25 and value= [User(id=1, name=費(fèi)爾南多, age=25)]
key= 26 and value= [User(id=2, name=費(fèi)爾南迪尼奧, age=26), User(id=5, name=隋維杰, age=26)]
key= 43 and value= [User(id=6, name=克魯伊夫, age=43)]
key= 27 and value= [User(id=3, name=卡爾德克, age=27)]
key= 28 and value= [User(id=4, name=阿德里安, age=28)]

結(jié)果是一個(gè)map,key為不重復(fù)的年齡,value為屬于該年齡的隊(duì)員列表。已經(jīng)實(shí)現(xiàn)了分組。另外我們還可以繼續(xù)分組得到兩次分組的結(jié)果。

2)如果僅僅想統(tǒng)計(jì)各年齡的隊(duì)員個(gè)數(shù)是多少,并不需要對(duì)應(yīng)的list

按年齡分組并統(tǒng)計(jì)人數(shù):

@Test
    public void testUserGroupingByCount(){
        List userList = userMapper.getALL();
        Map ageMap  = userList.stream().collect(Collectors.groupingBy(User::getAge,Collectors.counting()));
        for (Map.Entry entry :ageMap.entrySet()){
            System.out.println("隊(duì)員中" + entry.getKey() + "歲的隊(duì)員人數(shù)為:" + entry.getValue());
        }
    }

執(zhí)行結(jié)果為:

隊(duì)員中25歲的隊(duì)員人數(shù)為:1
隊(duì)員中26歲的隊(duì)員人數(shù)為:2
隊(duì)員中43歲的隊(duì)員人數(shù)為:1
隊(duì)員中27歲的隊(duì)員人數(shù)為:1
隊(duì)員中28歲的隊(duì)員人數(shù)為:1
3)partitioningBy 分區(qū)

分區(qū)與分組的區(qū)別在于,分區(qū)是按照 true 和 false 來(lái)分的,因此partitioningBy 接受的參數(shù)的 lambda 也是 T -> boolean

@Test
    public void testUserPartitioningBy (){
        List userList = userMapper.getALL();
        Map> partitioningByMap  = userList.stream().collect(partitioningBy(user -> user.getAge() >= 30));
        for (Map.Entry> entry :partitioningByMap.entrySet()){
            System.out.println("key= " + entry.getKey() + " and value= " + entry.getValue());
        }
    }

執(zhí)行結(jié)果為:

key= false and value= [User(id=1, name=費(fèi)爾南多, age=25), User(id=2, name=費(fèi)爾南迪尼奧, age=26), User(id=3, name=卡爾德克, age=27), User(id=4, name=阿德里安, age=28), User(id=5, name=隋維杰, age=26)]
key= true and value= [User(id=6, name=克魯伊夫, age=43)]

寫(xiě)在后面的話】留下stream的類實(shí)現(xiàn)的方法和依賴圖,前面的實(shí)踐也只是挑選了幾個(gè)比較常用的Api。

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/76086.html

相關(guān)文章

  • Java 8怎么了:局部套用vs閉包

    摘要:本文主要介紹了中的閉包與局部套用功能,由國(guó)內(nèi)管理平臺(tái)編譯呈現(xiàn)。譬如,認(rèn)為給帶來(lái)了閉包特性就是其中之一。但是首先,我們將考慮如何利用閉包進(jìn)行實(shí)現(xiàn)。很顯然,閉包打破了這一準(zhǔn)則。這就是局部調(diào)用,它總是比閉包更為穩(wěn)妥。 【編者按】本文作者為專注于自然語(yǔ)言處理多年的 Pierre-Yves Saumont,Pierre-Yves 著有30多本主講 Java 軟件開(kāi)發(fā)的書(shū)籍,自2008開(kāi)始供職于 ...

    3fuyu 評(píng)論0 收藏0
  • Java編程方法論:響應(yīng)式RxJava與代碼設(shè)計(jì)實(shí)戰(zhàn)》序

    摘要:原文鏈接編程方法論響應(yīng)式與代碼設(shè)計(jì)實(shí)戰(zhàn)序,來(lái)自于微信公眾號(hào)次靈均閣正文內(nèi)容在一月的架構(gòu)和設(shè)計(jì)趨勢(shì)報(bào)告中,響應(yīng)式編程和函數(shù)式仍舊編列在第一季度的早期采納者中。 原文鏈接:《Java編程方法論:響應(yīng)式RxJava與代碼設(shè)計(jì)實(shí)戰(zhàn)》序,來(lái)自于微信公眾號(hào):次靈均閣 正文內(nèi)容 在《2019 一月的InfoQ 架構(gòu)和設(shè)計(jì)趨勢(shì)報(bào)告》1中,響應(yīng)式編程(Reactive Programming)和函數(shù)式...

    PAMPANG 評(píng)論0 收藏0
  • [譯] Java 中最常見(jiàn)的 5 個(gè)錯(cuò)誤

    摘要:近日,在上列舉了開(kāi)發(fā)中常見(jiàn)的個(gè)錯(cuò)誤,與君共免。在多線程中并發(fā)修改集合內(nèi)容是非常常見(jiàn)的,因此需要使用并發(fā)編程中常用的方法進(jìn)行處理,例如同步鎖對(duì)于并發(fā)修改采用特殊的集合等等。在單線程和多線程情況下解決這個(gè)問(wèn)題有微小的差別。 在編程時(shí),開(kāi)發(fā)者經(jīng)常會(huì)遭遇各式各樣莫名錯(cuò)誤。近日,Sushil Das 在 Geek On Java上列舉了 Java 開(kāi)發(fā)中常見(jiàn)的 5 個(gè)錯(cuò)誤,與君共「免」。 原文...

    chemzqm 評(píng)論0 收藏0
  • Java8新特性總覽

    摘要:新特性總覽標(biāo)簽本文主要介紹的新特性,包括表達(dá)式方法引用流默認(rèn)方法組合式異步編程新的時(shí)間,等等各個(gè)方面。還有對(duì)應(yīng)的和類型的函數(shù)連接字符串廣義的歸約匯總起始值,映射方法,二元結(jié)合二元結(jié)合。使用并行流時(shí)要注意避免共享可變狀態(tài)。 Java8新特性總覽 標(biāo)簽: java [TOC] 本文主要介紹 Java 8 的新特性,包括 Lambda 表達(dá)式、方法引用、流(Stream API)、默認(rèn)方...

    mayaohua 評(píng)論0 收藏0
  • Java 8函數(shù)式編程》作者Richard Warbourton:Java的亮點(diǎn)不是語(yǔ)言本身

    摘要:根據(jù)對(duì)社區(qū)和新特性的深刻理解,他創(chuàng)作了函數(shù)式編程一書(shū)。問(wèn)你在倫敦社區(qū)的經(jīng)歷是否幫助你創(chuàng)作了函數(shù)式編程這本書(shū)絕對(duì)是這樣。我認(rèn)為引入函數(shù)式編程會(huì)為很多編程任務(wù)提供方便。問(wèn)之前的是面向?qū)ο蟮模F(xiàn)在全面支持函數(shù)式編程。 非商業(yè)轉(zhuǎn)載請(qǐng)注明作譯者、出處,并保留本文的原始鏈接:http://www.ituring.com.cn/article/199271 Richard Warburto...

    mzlogin 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<