溫馨提示×

溫馨提示×

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

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

java函數(shù)式編碼結(jié)構(gòu)及優(yōu)勢

發(fā)布時(shí)間:2021-09-01 16:05:06 來源:億速云 閱讀:149 作者:chen 欄目:編程語言

這篇文章主要講解了“java函數(shù)式編碼結(jié)構(gòu)及優(yōu)勢”,文中的講解內(nèi)容簡單清晰,易于學(xué)習(xí)與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學(xué)習(xí)“java函數(shù)式編碼結(jié)構(gòu)及優(yōu)勢”吧!

前言

當(dāng)垃圾回收成為主流時(shí),它消除了所有類別的難以調(diào)試的問題,使運(yùn)行時(shí)能夠?yàn)殚_發(fā)人員管理復(fù)雜的、容易出錯(cuò)的進(jìn)程。函數(shù)式編程旨在為您編寫的算法實(shí)現(xiàn)同樣的優(yōu)化,這樣您就可以從一個(gè)更高的抽象層面開展工作,同時(shí)運(yùn)行時(shí)執(zhí)行復(fù)雜的優(yōu)化。

Java 下一代語言并不都占用從命令式到函數(shù)式的語言頻譜的同一位置,但都展現(xiàn)出函數(shù)功能和習(xí)語。函數(shù)式編程技術(shù)有明確定義,但語言有時(shí)為相同的函數(shù)式概念使用不同的術(shù)語,使得我們很難看到相似之處。在本期文章中,我比較了 Scala、Groovy 和 Clojure 的函數(shù)式編碼風(fēng)格并討論了它們的優(yōu)勢。

命令式處理

我要首先探討一個(gè)常見問題及其命令式解決方案。假如給定一個(gè)名稱列表,其中一些名稱包含一個(gè)字符。系統(tǒng)會要求您在一個(gè)逗號分隔的字符串中返回名稱,該字符串中不包含單字母的名稱,每個(gè)名稱的首字母都大寫。實(shí)現(xiàn)該算法的 Java 代碼如清單 1 所示。

清單 1. 命令式處理

public class TheCompanyProcess {public String cleanNames(List<String> listOfNames) {StringBuilder result = new StringBuilder();for(int i = 0; i < listOfNames.size(); i++) {if (listOfNames.get(i).length() > 1) {result.append(capitalizeString(listOfNames.get(i))).append(",");}}return result.substring(0, result.length() - 1).toString();}public String capitalizeString(String s) {return s.substring(0, 1).toUpperCase() + s.substring(1, s.length());}}

由于您必須處理整個(gè)列表,解決清單 1 中問題最簡單的方式是使用一個(gè)命令式循環(huán)。對于每個(gè)名稱,都需要進(jìn)行檢查,確認(rèn)其長度是否大于 1,然后(如果長度大于 1)將首字母大寫的名稱附加到 result 字符串,并在后面加逗號。最終字符串中的最后一個(gè)名稱不應(yīng)包含逗號,所以我將它從最后返回值中移走。

在命令式編程中,建議您在較低級上別執(zhí)行操作。在 清單 1 中的 cleanNames() 方法中,我執(zhí)行了三個(gè)任務(wù):我篩選 列表以消除單字符,將列表中每個(gè)名稱的首字母變換 為大寫,然后將列表轉(zhuǎn)化 為一個(gè)字符串。在命令式語言中,我不得不為三個(gè)任務(wù)都使用同一低級機(jī)制(對列表進(jìn)行迭代)。函數(shù)式語言將篩選、變換和轉(zhuǎn)化視為常見操作,因此它們提供給您從不同視角解決問題的方式。

函數(shù)式處理

函數(shù)編程語言與命令式語言的問題分類方式不同。篩選、變換和轉(zhuǎn)化邏輯類別表現(xiàn)為函數(shù)。那些函數(shù)實(shí)現(xiàn)低級變換并依賴于開發(fā)人員來編寫作為參數(shù)傳遞的函數(shù),進(jìn)而定制函數(shù)的行為。我可以用偽代碼將 清單 1 中的問題概念化為:

listOfEmps -> filter(x.length > 1) -> transform(x.capitalize) -> convert(x, y -> x + "," + y)

利用函數(shù)式語言,您可以建模這一概念性解決方案,無需擔(dān)心實(shí)現(xiàn)細(xì)節(jié)。

Scala 實(shí)現(xiàn)

清單 2 使用 Scala 實(shí)現(xiàn) 清單 1 中的處理示例。它看起來就像是前面的偽代碼,包含必要的實(shí)現(xiàn)細(xì)節(jié)。

清單 2. Scala 處理

val employees = List("neal", "s", "stu", "j", "rich", "bob")val result = employees.filter(_.length() > 1).map(_.capitalize).reduce(_ + "," + _)

對于給定的名稱列表,我首先篩選它,剔除長度不大于 1 的所有名稱。然后將該操作的輸出提供給 map() 函數(shù),該函數(shù)對集合的每個(gè)元素執(zhí)行所提供的代碼塊,返回變換后的集合。最后,來自 map() 的輸出集合流向 reduce() 函數(shù),該函數(shù)基于代碼塊中提供的規(guī)則將每個(gè)元素結(jié)合起來。

在本例中,我將每對元素結(jié)合起來,用插入的逗號連接它們。我不必考慮三個(gè)函數(shù)調(diào)用中參數(shù)的名稱是什么,所以我可以使用方便的 Scala 快捷方式,也就是說,使用 _ 跳過名稱。reduce() 函數(shù)從前兩個(gè)元素入手,將它們結(jié)合成一個(gè)元素,成為下一個(gè)串接中的第一個(gè)元素。在 “瀏覽” 列表的同時(shí),reduce() 構(gòu)建了所需的逗號分隔的字符串。

我首先展示 Scala 實(shí)現(xiàn)是因?yàn)槲覍λ恼Z法比較熟悉,而且 Scala 分別為篩選、變換和轉(zhuǎn)化概念使用了行業(yè)通用的名稱,即 filter、map 和 reduce。

Groovy 實(shí)現(xiàn)

Groovy 擁有相同的功能,但對它們進(jìn)行命名的方式與腳本語言(比如 Ruby)更加一致。清單 1 中處理示例的 Groovy 版本如清單 3 所示。

清單 3. Groovy 處理

class TheCompanyProcess {public static String cleanUpNames(List listOfNames) {listOfNames.findAll {it.length() > 1}.collect {it.capitalize()}.join(',')}}

盡管清單 3 在結(jié)構(gòu)上類似于 清單 2 中的 Scala 示例,但方法名稱不同。Groovy 的 findAll 集合方法應(yīng)用所提供的代碼塊,保留代碼塊為 true 的元素。如同 Scala,Groovy 包含一個(gè)隱式參數(shù)機(jī)制,為單參數(shù)代碼塊使用預(yù)定義的 it 隱式參數(shù)。collect 方法(Groovy 的 map 版本)對集合的每個(gè)元素執(zhí)行所提供的代碼塊。Groovy 提供一個(gè)函數(shù) (join()),使用所提供的分隔符將字符串集合串聯(lián)為單一字符串,這正是本示例中所需要的。

Clojure 實(shí)現(xiàn)

Clojure 是一個(gè)使用 reduce、map 和 filter 函數(shù)名的函數(shù)式語言,如清單 4 所示。

清單 4. Clojure 處理示例

(defn process [list-of-emps](reduce str (interpose "," (map clojure.string/capitalize (filter #(< 1 (count %)) list-of-emps)))))

Clojure 的 thread-first 宏

thread-last 宏 使集合的處理變得更加簡單。類似的 Clojure 宏 thread-first 可簡化與 Java API 的交互。例如普遍的 Java 代碼語句 person.getInformation().

getAddress().getPostalCode(),這體現(xiàn)了 Java 違反 迪米特法則 的傾向。這種類型的語句給 Clojure 編程帶來一些煩惱,迫使使用 Java API 的開發(fā)人員不得不構(gòu)建由內(nèi)而外的語句,比如 (getPostalCode (getAddress (getInformation person)))。thread-first 宏消除了這一語法困擾。您可以使用宏將嵌套調(diào)用編寫為 (-> person getInformation getAddress getPostalCode),想嵌套多少層都可以。

如果您不習(xí)慣查看 Clojure,可以使用清單 4 中的代碼,其結(jié)構(gòu)可能不夠清晰。Clojure 這樣的 Lisp 是 “由內(nèi)而外” 進(jìn)行工作的,所以必須從最后的參數(shù)值 list-of-emps 著手。Clojure 的 (filter ) 函數(shù)接受兩個(gè)參數(shù):用于進(jìn)行篩選的函數(shù)(本例中為匿名函數(shù))和要篩選的集合。

您可以為第一個(gè)參數(shù)編寫一個(gè)正式函數(shù)定義,比如 (fn [x] (< 1 (count x))),但使用 Clojure 可以更簡潔地編寫匿名函數(shù)。與前面的示例一樣,篩選操作的結(jié)果是一個(gè)較少的集合。(map ) 函數(shù)將變換函數(shù)接受為第一個(gè)參數(shù),將集合(本例中是 (filter ) 操作的返回值)作為第二個(gè)參數(shù)。Clojure 的 (map ) 函數(shù)的第一個(gè)參數(shù)通常是開發(fā)人員提供的函數(shù),但接受單一參數(shù)的任何函數(shù)都有效;內(nèi)置 capitalize 函數(shù)也符合要求。

最后,(map ) 操作的結(jié)果成為了 (reduce ) 的集合參數(shù)。(reduce ) 的第一個(gè)參數(shù)是組合函數(shù)(應(yīng)用于 (interpose ) 的返回的 (str ))。(interpose ) 在集合的每個(gè)元素之間(除了最后一個(gè))插入其第一個(gè)參數(shù)。

當(dāng)函數(shù)嵌套過多時(shí),即使最有經(jīng)驗(yàn)的開發(fā)人員也會倍感頭疼,如 清單 4 中的 (process ) 函數(shù)所示。所幸的是,Clojure 包含的宏支持您將結(jié)構(gòu) “調(diào)整” 為更可讀的順序。清單 5 中的功能與 清單 4 中的功能一樣。

清單 5. 使用 Clojure 的 thread-last 宏

(defn process2 [list-of-emps](->> list-of-emps(filter #(< 1 (count %)))(map clojure.string/capitalize)(interpose ",")(reduce str)))

Clojure thread-last 宏采取對集合應(yīng)用各種變換的常見操作并顛倒典型的 Lisp 的順序,恢復(fù)了從左到右的更自然的閱讀方式。在 清單 5 中,首先是 (list-of-emps) 集合。代碼塊中每個(gè)隨后的表單被應(yīng)用于前一個(gè)表單。Lisp 的優(yōu)勢之一在于其語法靈活性:任何時(shí)候代碼的可讀性變得很差時(shí),您都可以將代碼調(diào)整回具有較高可讀性。

函數(shù)式編程的優(yōu)勢

在一篇標(biāo)題為 “Beating the Averages” 的著名文章中,Paul Graham 定義了 Blub Paradox:他 “編造” 了一種名為 Blub 的虛假語言,并且考慮在其他語言與 Blub 之間進(jìn)行功能比較:

只要我們假想的 Blub 程序員往下看一連串功能,他就知道自己是在往下看。不如 Blub 功能強(qiáng)大的語言顯然不怎么強(qiáng)大,因?yàn)樗鼈內(nèi)鄙俪绦騿T習(xí)慣使用的一些功能。但當(dāng)我們假想的 Blub 程序員從另一個(gè)方向,也就是說,往上看一連串功能時(shí),他并沒有意識到自己在往上看。他看到的只不過是怪異的語言。他可能認(rèn)為它們在功能上與 Blub 幾近相同,只是多了其他難以理解的東西。Blub 對他而言已經(jīng)足夠好,因?yàn)樗窃?Blub 環(huán)境中可以思考問題。

對于很多 Java 開發(fā)人員而言,清單 2 中的代碼看起來陌生而又奇怪,因此難以將它看作是有優(yōu)勢的代碼。但當(dāng)您停止過于細(xì)化任務(wù)執(zhí)行細(xì)節(jié)時(shí),就釋放了越來越智能的語言和運(yùn)行時(shí)的潛能,從而做出了強(qiáng)大的改進(jìn)。例如,JVM 的到來(解除了開發(fā)人員的內(nèi)存管理困擾)為先進(jìn)垃圾回收的創(chuàng)建開辟了全新的研發(fā)領(lǐng)域。使用命令式編碼時(shí),您深陷于迭代循環(huán)的細(xì)節(jié),難以進(jìn)行并行性等優(yōu)化。從更高的層面思考操作(比如 filter、map 和 reduce)可將概念與實(shí)現(xiàn)分離開來,將并行性等修改從一項(xiàng)復(fù)雜、詳細(xì)的任務(wù)轉(zhuǎn)變?yōu)橐粋€(gè)簡單的 API 更改。

想一想如何將 清單 1 中的代碼變?yōu)槎嗑€程代碼。由于您密切參與了 for 循環(huán)期間發(fā)生的細(xì)節(jié),所以您還必須處理煩人的并發(fā)代碼。然后思考一下清單 6 所示的 Scala 并行版本。

清單 6. 實(shí)現(xiàn)進(jìn)程并行性

val parallelResult = employees.par.filter(f => f.length() > 1).map(f => f.capitalize).reduce(_ + "," + _)

清單 2 與 清單 6 之間惟一的差別在于,將 .par 方法添加到了命令流中。.par 方法返回后續(xù)操作依據(jù)的集合的并行版本。由于我將對集合的操作指定為高階概念,所以底層運(yùn)行時(shí)可以自由地完成更多的工作。

面向命令式對象的開發(fā)人員往往會考慮使用重用類,因?yàn)樗麄兊恼Z言鼓勵(lì)將類作為構(gòu)建塊。函數(shù)編程語言傾向于重用函數(shù)。函數(shù)式語言構(gòu)建復(fù)雜的通用功能(比如 filter()、map() 和 reduce())并通過作為參數(shù)提供的函數(shù)來實(shí)現(xiàn)定制。在函數(shù)式語言中,將數(shù)據(jù)結(jié)構(gòu)轉(zhuǎn)換為列表和映射等標(biāo)準(zhǔn)集合是很尋常的事,因?yàn)樗鼈兘又涂梢员粡?qiáng)大的內(nèi)置函數(shù)所操控。

例如,在 Java 環(huán)境中存在許多 XML 處理框架,每個(gè)框架都封裝自己的私有版本的 XML 結(jié)構(gòu),并通過自己的方法交付它。在 Clojure 這樣的語言中,XML 被轉(zhuǎn)換為基于映射的標(biāo)準(zhǔn)數(shù)據(jù)結(jié)構(gòu),該結(jié)構(gòu)對已經(jīng)存在于語言中的強(qiáng)大的變換、約簡和篩選操作開放。

結(jié)束語

所有現(xiàn)代語言都包含或添加了函數(shù)式編程結(jié)構(gòu),使函數(shù)式編程成為未來開發(fā)中不可或缺的一部分。Java 下一代語言都實(shí)現(xiàn)了強(qiáng)大的函數(shù)式功能,有時(shí)使用不同的名稱和行為。

感謝各位的閱讀,以上就是“java函數(shù)式編碼結(jié)構(gòu)及優(yōu)勢”的內(nèi)容了,經(jīng)過本文的學(xué)習(xí)后,相信大家對java函數(shù)式編碼結(jié)構(gòu)及優(yōu)勢這一問題有了更深刻的體會,具體使用情況還需要大家實(shí)踐驗(yàn)證。這里是億速云,小編將為大家推送更多相關(guān)知識點(diǎn)的文章,歡迎關(guā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