Java8函數(shù)式編程

在上一篇文章中蹄皱,我們大致介紹了函數(shù)式編程的情況,包括函數(shù)式編程的緣起芯肤,意義巷折,還有高階函數(shù),柯里化等特點崖咨,今天我們將從lambda風(fēng)格锻拘,流操作,高階函數(shù)击蹲,并行化等幾個方面署拟,一起簡單聊聊函數(shù)式編程在Java8中的實現(xiàn)。

Lambda風(fēng)格

在上篇介紹函數(shù)式編程的文章中歌豺,我們大致了解了lambda表達式的來源以及定義推穷,那么lambda表達式在java8中是怎么使用的呢?
lambda表達式允許函數(shù)作為一個方法的參數(shù)类咧,在代碼的實現(xiàn)上顯得更加簡潔緊湊馒铃。
lambda表達式的格式語法如下

(parameters) -> expression

(parameters) -> {statement;}

在lambda表達式出現(xiàn)之前,如果想將代碼作為數(shù)據(jù)傳遞痕惋,一般會采用匿名類的方式区宇。比如在swing編程中,如果想在button上添加點擊操作對應(yīng)的事件監(jiān)聽器

 botton.addActionListener(new ActionListener(){
     public void actionPerformed(ActionEvent event){
         // Print something
     }
 }
 )值戳;

如果使用lambda表達式议谷,可以一行代碼解決

botton.addActionListener(event -> System.out.println("something"));

使用匿名類需要顯式聲明參數(shù)類型ActionEvent,但是會使用lambda表達式無需指定類型堕虹,這是由于javac根據(jù)程序的上下文在后臺推斷出event的類型

流操作

流操作中包括惰性求值和及早求值兩種卧晓,一般來說,對于只描述stream,不產(chǎn)生新集合的方法我們稱為惰性求值赴捞;而對于最終會從stream產(chǎn)生值的方法逼裆,我們稱為及早求值。常見的惰性求值有map,reduce,filter等螟炫,及早求值包括count,sum波附,collect等艺晴。
下面昼钻,我們采用漸進式重構(gòu)的方式來舉例介紹常用的幾個流操作掸屡。

  • 命令式編程
public Set<String> findLongTracks(List<Albulm>){
    Set<String> trackNames = new HashSet<>(); 
    for(Album album : albums) {
        for (Track track : album.getTrackList()) { if (track.getLength() > 60) {
                String name = track.getName();
                trackNames.add(name);
            }
        }
    }
    return trackNames;
}
  • stream化,并使用foreach進行重構(gòu)
  public Set<String> findLongTracks(List<Albulm>){
    Set<String> trackNames = new HashSet<>(); 
    albums.stream().
        forEach(album -> {
                   album.getTracks().forEach(track -> {
                        if (track.getLength() > 60) {
                             String name = track.getName();
                             trackNames.add(name);
                        }
                    );
                 }
            );
    return trackNames; 
}  
  • 使用map,filter進行重構(gòu)
public Set<String> findLongTracks(List<Album> albums) { 
    Set<String> trackNames = new HashSet<>(); 
    albums.stream()
               .forEach(album -> {
                   album.getTracks()
                        .filter(track -> track.getLength() > 60)
                        .map(track -> track.getName())
                        .forEach(name -> trackNames.add(name));
                });
    return trackNames;
}
  • 使用map,collect繼續(xù)進行重構(gòu)
public Set<String> findLongTracks(List<Album> albums) { 
    Set<String> trackNames = new HashSet<>(); 
    albums.stream()
               .map(album -> {
                   album.getTracks()
                        .filter(track -> track.getLength() > 60)
                        .map(track -> track.getName())
                        .forEach(name -> trackNames.add(name));
                }).collect(Collecotor.toSet());
    return trackNames;
}    
  • 使用flatMap進行重構(gòu)
public Set<String> findLongTracks(List<Album> albums) { 
    return albums.stream()
                .flatMap(album -> album.getTracks())
                .filter(track -> track.getLength() > 60)
                .map(track -> track.getName())
                .collect(toSet());
}

高階函數(shù)

高階函數(shù)是使用其他函數(shù)作為參數(shù)然评,并返回其他函數(shù)作為結(jié)果仅财。它可以幫助我們將創(chuàng)建和處理區(qū)分開來,通過這種方式碗淌,新的業(yè)務(wù)邏輯處理對象就可以輕易的添加進來盏求,而沒有必要同對象創(chuàng)建邏輯相耦合。

public interface Block<T> {
    void apply(T t);
}

public void doWithContact(String fileName, Block<Contact> block) {
    try {
        String contacStr = FileUtils.readFileToString(new File(fileName));
        Contact.apply(contact);
        block.apply(contact);
    }
    catch (IOException e) {
        System.out.println("cloudn't load contact file: " + e.getMessage());
    }   
    catch (ParseException e) {
        System.out.println("cloudn't parse contact file: " + e.getMessage());
    }
}

//usage
doWithContact("custerX.vcf", c -> ContactDao.save(c))

并行化

并發(fā)(Concurrent)與并行(Parallel)

這是兩個比較容易混淆的概念亿眠。二者都可以表示兩個或者多個任務(wù)一起執(zhí)行碎罚,但是側(cè)重點不同;并發(fā)側(cè)重于多個任務(wù)交替執(zhí)行纳像,而多個任務(wù)之間有可能是串行的荆烈;并發(fā)是邏輯上的同時發(fā)生,而并行是物理上的同時發(fā)生竟趾。嚴(yán)格意義上憔购,并行的多個任務(wù)是真實同時執(zhí)行,而對于并發(fā)岔帽,則是交替的玫鸟,一會執(zhí)行任務(wù)A,一會執(zhí)行任務(wù)B犀勒,系統(tǒng)會不斷地在兩者之間進行切換屎飘,

并行化操作流

并行化操作流只需要改變一個方法的調(diào)用,如果已經(jīng)存在y一個stream對象贾费,調(diào)用它的parallel方法就能讓其立即擁有并行操作的能力枚碗。

public Set<String> findLongTracks(List<Album> albums) { 
    return albums.parallelStream()
                .flatMap(album -> album.getTracks())
                .filter(track -> track.getLength() > 60)
                .map(track -> track.getName())
                .collect(toSet());
}

parallelStream的底層與Fork/Join類似,默認(rèn)的并發(fā)程度是可用CPU數(shù)-1
在使用并行化編程時铸本,需要注意:

  1. 線程安全
  2. 合理參數(shù)配置:比如線程池的大小肮雨,等待隊列大小,并行度大小以及等待超時時間等等箱玷,都需要根據(jù)自己的業(yè)務(wù)不斷的調(diào)優(yōu)防止出現(xiàn)隊列不夠用或者超時時間不合理等
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末怨规,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子锡足,更是在濱河造成了極大的恐慌波丰,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,509評論 6 504
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件舶得,死亡現(xiàn)場離奇詭異掰烟,居然都是意外死亡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,806評論 3 394
  • 文/潘曉璐 我一進店門纫骑,熙熙樓的掌柜王于貴愁眉苦臉地迎上來蝎亚,“玉大人,你說我怎么就攤上這事先馆》⒖颍” “怎么了?”我有些...
    開封第一講書人閱讀 163,875評論 0 354
  • 文/不壞的土叔 我叫張陵煤墙,是天一觀的道長梅惯。 經(jīng)常有香客問我,道長仿野,這世上最難降的妖魔是什么铣减? 我笑而不...
    開封第一講書人閱讀 58,441評論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮脚作,結(jié)果婚禮上徙歼,老公的妹妹穿的比我還像新娘。我一直安慰自己鳖枕,他們只是感情好魄梯,可當(dāng)我...
    茶點故事閱讀 67,488評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著宾符,像睡著了一般酿秸。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上魏烫,一...
    開封第一講書人閱讀 51,365評論 1 302
  • 那天辣苏,我揣著相機與錄音,去河邊找鬼哄褒。 笑死稀蟋,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的呐赡。 我是一名探鬼主播退客,決...
    沈念sama閱讀 40,190評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼链嘀!你這毒婦竟也來了萌狂?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,062評論 0 276
  • 序言:老撾萬榮一對情侶失蹤怀泊,失蹤者是張志新(化名)和其女友劉穎茫藏,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體霹琼,經(jīng)...
    沈念sama閱讀 45,500評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡务傲,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,706評論 3 335
  • 正文 我和宋清朗相戀三年凉当,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片售葡。...
    茶點故事閱讀 39,834評論 1 347
  • 序言:一個原本活蹦亂跳的男人離奇死亡看杭,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出天通,到底是詐尸還是另有隱情,我是刑警寧澤熄驼,帶...
    沈念sama閱讀 35,559評論 5 345
  • 正文 年R本政府宣布像寒,位于F島的核電站,受9級特大地震影響瓜贾,放射性物質(zhì)發(fā)生泄漏诺祸。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,167評論 3 328
  • 文/蒙蒙 一祭芦、第九天 我趴在偏房一處隱蔽的房頂上張望筷笨。 院中可真熱鬧,春花似錦龟劲、人聲如沸胃夏。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,779評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽仰禀。三九已至,卻和暖如春蚕愤,著一層夾襖步出監(jiān)牢的瞬間答恶,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,912評論 1 269
  • 我被黑心中介騙來泰國打工萍诱, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留悬嗓,地道東北人。 一個月前我還...
    沈念sama閱讀 47,958評論 2 370
  • 正文 我出身青樓裕坊,卻偏偏與公主長得像包竹,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子籍凝,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,779評論 2 354

推薦閱讀更多精彩內(nèi)容