從Python到Haskell:程式設計師為何與函數式編程「墜入愛河」?

2020-12-18 讀芯術

全文共3588字,預計學習時長9分鐘

圖源:unsplash

函數式編程發展至今已有60年的歷史,但是截至目前,它仍然算是比較小眾。儘管像Google這樣的大公司依賴於函數式編程的關鍵概念,但是普通程式設計師對此幾乎一無所知。

這種情況即將改變了。不僅是Java或Python這樣的語言越來越多地採用了函數式編程的概念,類似Haskell這樣的新語言也正在完全實現函數式編程。

簡單來說,函數式編程就是為不可變變量構建函數。與之相反,面向對象的編程則是有一組相對固定的函數,而用戶主要是修改或添加新變量。

由於函數式編程的特性,它非常適合完成諸如數據分析和機器學習之類的需求任務。但是這並不意味著用戶要告別面向對象的編程,轉而完全使用函數式編程。但用戶需要了解其基本原理,以便在適當的時候使用它們以發揮優勢。

一切都是為了消除副作用

要了解函數式編程,首先需要了解函數。函數是將輸入轉換為輸出的東西,它並不總是這麼簡單。下面看一個Python中的函數:

def square(x):

return x*x

這個函數很簡單。它需要一個變量 x,或者是一個int,又或者是float或double,然後輸出該變量的平方。

現在再思考這個函數:

lobal_list = []def append_to_list(x):

global_list.append(x)

乍一看,該函數看起來像是接受了一個任意類型的變量x,並且由於沒有 return 語句,它不會返回任何值。

請等一下!如果未事先定義global_list,那麼該函數將不起作用,並且在經過修改後仍輸出相同的列表。儘管global_list從未被視為函數的輸入,但使用函數時它也會發生改變:

append_to_list(1)

append_to_list(2)

global_list

它將返回[1,2]而不是一個空列表。即使我們對此並不明確,但這表明該列表確實是該函數的輸入。這種不明確可能會造成問題。

圖源:GitHub

不忠實於函數

這些隱含的輸入,或在其他情況下的輸出,有一個官方的名稱:side effects(副作用)。雖然本文所舉的只是一個簡單的示例,但是在更複雜的程序中,這些副作用可能會導致真正的困難。

請思考一下如何測試append_to_list:用戶不僅需要閱讀第一行並使用任意的x來測試函數,還需要閱讀整個定義,理解其作用,定義global_list並且以這種方式進行測試。當需要處理帶有數千行代碼的程序時,此示例中的簡單操作可能很快就會變得乏味無趣。

有一個簡單的解決方法:忠於函數認定為輸入的內容。

newlist = []def append_to_list2(x, some_list):

some_list.append(x)append_to_list2(1,newlist)

append_to_list2(2,newlist)

newlist

它並沒有做出太大的改變。輸出仍然是[1,2],並且其他所有內容也保持不變。但是有一樣改變了:該代碼現在擺脫了副作用。

現在,當查看函數聲明時,用戶能確切地知道發生了什麼。因此,如果程序運行不正常,用戶也可以輕而易舉地單獨測試每個功能,並查明哪個功能有問題

函數式編程正在編寫純函數

沒有副作用的函數是指其輸入和輸出都具有明確的聲明,而沒有副作用的功能就是純函數。

函數式編程一個非常簡單的定義:僅用純函數編寫程序。純函數永遠不會修改變量,而只會創建新的變量作為輸出。(筆者在上面的示例中稍微「作弊」了一下:它遵循函數式編程的原則,但仍使用全局列表。用戶可以找到更好的示例,但這只是基本原則。)

此外,對於給定輸入的純函數,可以得到特定的輸出。相反,不純函數則依賴於一些全局變量。因此,如果全局變量不同,則相同的輸入變量可能導致不同的輸出。不純函數會使代碼的調試和維護變得更加困難。

有一個更容易發現副作用的小竅門:由於每個函數都必須具有某種輸入和輸出,因此沒有任何輸入或輸出的函數聲明一定是不純的。如果採用函數式編程,這些則可能是第一批需要的更改聲明。

圖源:unsplash

函數式編程不僅只有Map和reduce

函數式編程中不包含循環結構(Loops),請看下面這些Python中的循環:

integers = [1,2,3,4,5,6]

odd_ints = []

squared_odds = []

total = 0for i in integers:

if i%2 ==1

odd_ints.append(i)for i inodd_ints:

squared_odds.append(i*i)for i insquared_odds:

total += i

相較於我們要執行的簡單操作,以上代碼明顯過長。而且由於修改全局變量,它也不夠有效。我們可以用以下代碼替代:

from functools import reduceintegers = [1,2,3,4,5,6]

odd_ints = filter(lambda n: n % 2 == 1, integers)

squared_odds = map(lambda n: n * n, odd_ints)

total = reduce(lambda acc, n: acc + n, squared_odds)

這是完整的函數。因為不需要迭代一個數組的許多元素,所以它更短也更快。而且,一旦了解了 filter、map和reduce 如何工作,代碼也就容易理解了。但這並不意味著所有函數代碼都使用map、reduce 等。這也不意味著需要藉助函數式編程來理解map 和 reduce,這些函數只是在抽象循環時彈出很多。

· Lambda functions:談及函數式編程的發展史時,許多人都會先提及lambda函數的發明。儘管,lambda毫無疑問是函數式編程的基石,但這並不是根本原因。Lambda函數是可使程序發揮作用的工具。但是,lambda也可用於面向對象的編程。

· Static typing:上面的示例不屬於靜態輸入,而是函數式的。即使靜態類型為代碼增加了一層額外的安全保護,但也並非一定要其函數化,不過這可能會是錦上添花。

一些語言對函數式編程更加友好

圖源:unsplash

Perl

Perl對於副作用的處理方法與大多數程式語言截然不同。它包含一個神奇的參數 $_,這使得處理副作用成為Perl核心功能之一。儘管Perl確實有其優點,但作者不會嘗試使用它進行函數式編程。

Java

如果要用Java編寫函數式代碼的話,只能自求多福了。因為該程序的一半不僅將都是static 關鍵字,而且其他大多數Java開發人員也會將此程序視為恥辱。

Scala

Scala是一個很有趣的語言:它的目標是統一面向對象和函數式編程。很多人都覺得這很奇怪,因為函數式編程旨在徹底消除副作用,而面向對象的編程則試圖將副作用保留在對象內部。

話雖如此,許多開發人員將Scala視為一種可以幫助他們從面向對象編程過渡到函數式程式語言,這可能會幫助他們在未來幾年更容易完全過渡到函數式編程。

Python

Python積極鼓勵使用函數式編程。下列事實證明了這一點:每個函數在默認情況下都有至少有一個輸入self。這就像是Python之禪:顯式比隱式好!

Clojure

根據其創建者的說法,Clojure的函數化達到80%。默認情況下,正如在函數式編程中所需要的,它的所有值都是不可變的。但是,可以通過對這些不可變值使用可變值包裝類來解決此問題。當打開這樣的包裝類,可變值將再次不可變。

Haskell

這是極少數純函數式和靜態類型的語言之一。儘管在開發過程中可能會耗費大量時間,但在調試程序時這些付出都會獲得巨大回報。它不像其他語言那樣容易學習,但是絕對值得花時間學習。

圖源:unsplash

與面向對象的編程相比,函數式編程仍然小眾。但是,如果說在Python和其他語言中加入函數式編程原理意味著什麼的話,那就是函數式編程正越來越受到關注。這完全說得通:函數式編程對於大型資料庫、並行編程和機器學習大有裨益。而在過去十年間,這些迎來了蓬勃發展。

雖然面向對象編程有著不可估量的優點,但函數代碼的優點也不容忽視。只需要學習一些基本原理,就足以讓用戶成為一名開發人員,並為未來做好準備。

留言點讚關注

我們一起分享AI學習與發展的乾貨

如轉載,請後臺留言,遵守轉載規範

相關焦點

  • 寫Python 代碼不可不知的函數式編程技術
    選自 Medium作者:Raivat Shah參與:魔王、Jamin本文對 Python 中的函數式編程技術進行了簡單的入門介紹。近來,越來越多人使用函數式編程(functional programming)。因此,很多傳統的命令式語言(如 Java 和 Python)開始支持函數式編程技術。本文對 Python 中的函數式編程技術進行了簡單的入門介紹。本文適合對函數式編程有基本了解的讀者。
  • 輕鬆玩轉函數式編程
    編程範式 函數式編程 函數式編程常見案例 編程範式 編程範式指的是一種編程風格,它描述了程式設計師對程序執行的看法。
  • 《python 入陣曲:初級》開題報告
    看那些python入門視頻的評論區,有寫「第五次入門,希望堅持到AI部分」的評論,讓人覺得很真實也很遺憾,有這麼多喜歡學習編程的人到處求學,卻沒有好好把編程這門學問低門檻地科普給愛好者的視頻和作者。學術學派恨不得你把所有程序都用彙編來寫,寫出來的程序都是模板套類套設計模式;而工具學派恨不得你把程序當作工具箱,把程式設計師當技工,培訓即刻上崗。這樣學出來的程式設計師,多數不是書呆子就是半吊子,我就想搞一個「有情懷的編程課程」,實戰與理論結合、方法與思想並重,把筆者這些年對編程的領悟講解給各位編程愛好者。
  • python高階函數:map、filter、reduce的替代品
    什麼是高階函數?高階函數是一種將函數作為參數,或者把函數作為結果返回的函數,map函數、sorted函數就是高階函數的典型例子。map函數在小編以前的文章中做過相應的知識分享。sorted函數是python的內置函數,它的可選參數key用於提供一個函數,它可以將函數應用到各個元素上進行排序。
  • 代碼森林帶你實踐熱門編程書籍——《Python編程:從入門到實踐》
    第一本你必須要擁有的書籍一定是《Python編程:從入門到實踐》它是一本針對所有層次的Python讀者的編程學習書——《Python編程:從入門到實踐》本書內容本書旨在讓你成為優秀的程式設計師,具體地說,是優秀的Python程式設計師。
  • 函數式編程那些事兒
    函數式編程是一種編程範式,在其中它試圖將每個函數都綁定到純數學函數中。這是一種聲明式的編程風格,著重於解決什麼而不是如何解決。Clojure,Common Lisp,Erlang,Haskell和Scala是遵循函數式編程方法的一些著名程式語言。
  • Python 初學者進階的九大技能
    初學者與中級程式設計師那麼,對於Python程式設計師而言,初學者和進階者有什麼區別呢?使用列表;使用循環;使用函數(並正確談論函數);面向對象編程;尊重PEP。不理解為何代碼不運行的情況總會發生,當進行故障排除並搞清楚其原因時,思考代碼不運行的原因和最終使其運行的因素非常重要。這次學到的知識會帶到下一個程序中。例如,如果多個縮進級別的代碼中出現了縮進錯誤,可以嘗試隨機調整代碼塊,然後在最終運行時為自己慶祝。
  • 大數據入門:Scala函數式編程
    在大數據的學習當中,學到Spark部分,就免不了需要Scala這門語言了,Scala是Spark框架的原生程式語言,想要真正把Spark搞懂,那麼對Scala自然也需要相應程度的掌握。今天的大數據入門分享,我們就來講講Scala函數式編程。
  • java8的函數式編程解析
    其實在java8就已經有java的函數式編程寫法,只是難度較大,大家都習慣了對象式用法,但在其它語言中都有函數式的用法,如js,scala,函數式其實是抽象到極致的思想。什麼是函數式編程 函數式編程並不是Java新提出的概念,其與指令編程相比,強調函數的計算比指令的計算更重要;與過程化編程相比,其中函數的計算可以隨時調用。
  • 北大青鳥:2020年5月中國程式語言排行榜,哪種編程工資最高呢?
    :0.71% 簡介:Scala是一門多範式的程式語言,一種類似java的程式語言 ,設計初衷是實現可伸縮的語言 、併集成面向對象編程和函數式編程的各種特性。:0.10% 簡介:Rust是一門系統程式語言 ,專注於安全 ,尤其是並發安全,支持函數式和命令式以及泛型等編程範式的多範式語言。
  • python與c語言的語法有哪些不一樣的
    在眾多程式語言之中,想必很多人都聽說過Python和C語言,在進行編程學習之前,大家都會問:python和c語言的區別有哪些?我該如何選擇?接下來我們來看看吧。python與C的區別如下:1、語言類型:Python是一種基於解釋器的語言,會逐行讀取代碼,將Python編譯為字節碼,由大型C程序解釋;C是一種編譯語言,完整的原始碼將直接編譯為機器代碼,由CPU直接執行。
  • 知識分享之函數式編程的簡單介紹
    各位小夥伴們大家好,好久不見,這次小編要分享的一個知識是有關於函數式編程的。函數式編程是近年來比較熱門的一個話題,很多人都在談FunctionalProgramming,函數式編程有如下特點:函數即為數據,第一等公民。
  • 寫出漂亮 Python 代碼的 20條準則
    然而,當我們必須花大把時間來理解一個人的隱式代碼時,這項工作肯定不受歡迎,這種情況同樣可能發生在別人閱讀我們的代碼時。所以,讓我們聚焦 Python 之禪和一些改進技巧,從而解決問題。在這篇文章中,我將分享自己對這些格言的理解以及我學到的一些有用的 Python 技巧。Python 具有語法簡單、代碼可讀性強和命令類似英語等特點,這讓編寫 Python 代碼比使用其他程式語言更容易、更高效。
  • 函數式編程很難懂?其實真心很簡單
    話不多說,開始今天的學習:現在直播一直都很火,今天我們就用Java代碼簡單地模擬一個直播案例,以此來一步步說明什麼叫函數式編程。不要看這個名字好像挺難懂的樣子,其實很簡單,兩分鐘時間即可看完。可以的,也就是今天的重點,函數式編程。二、函數式編程函數,這個概念我們在數學裡面我們就接觸過。y=f(x)(y=x+1)這就是函數的格式,其中f是函數名,x是變量,y是函數值,還有定義域,值域什麼的。
  • 新手請進:每個Python程式設計師都應該知道的10個縮寫詞
    OOP(面向對象編程)要介紹的第一個縮寫是OOP——面向對象編程,這就是Python所基於的設計。大家都知道編程本身是關於編碼的,但是程序本身應該是關於數據的。程序需要獲取輸入數據、處理數據和輸出數據。
  • 開始你的編程之旅吧!從Python入門講起……
    但是,我相信你會認識到編程是一門自然的語言。Python程式語言世界上有700多種程式語言可供選擇,人們只注意到了其中一部分。我從2015年開始學習編程,一開始接觸的是C語言,我現在也對C語言心懷敬畏。但是,在2019年學習Python時,我第一眼就喜歡上了它,現在依舊如此。
  • 如何自學成 Python 大神?這裡有些建議
    如果你真的想要高效地學習 Python,那就需要掌握如何過濾網上的垃圾教程資源的技能,並從其他有經驗的程式設計師或在線編程社區中獲得幫助。倘若你沒有任何編程知識,或者知之甚少,從一張白紙起步,或許會更容易一些。
  • 資料|像計算機科學家一樣思考Python(中文版)
    from=leiphonecolumn_res0903內容簡介 · · · · · ·《像計算機科學家一樣思考python》按照培養讀者像計算機科學家一樣的思維方式的思路來教授python語言編程。全書貫穿的主體是如何思考、設計、開發的方法,而具體的程式語言,只是提供一個具體場景方便介紹的媒介。
  • 一周學全Python面試基礎(2)
    通過列出30個python面試問題和答案,本文涵蓋在Python面試中經常問到的問題。如果您是該行業的新手,本基礎篇將極大地幫助您。我們衷心希望這篇文章在準備面試時會有所幫助。Python的需求量很大,必須與成千上萬擁有與python技能的申請人競爭,才能在就業市場中找到工作。
  • 15課python快遞編程代碼人Python語法簡單才會越來越被編程界歡迎
    分析流程,拆解項目我們的任務就是做一個「快遞配送調配程序」,雖然這朋友要的比較著急,但是我們也不著急於編程開發,應該先梳理清楚需求,做出的程序功能達到什麼樣的效果。計算器,首先配送站BOSS要輸入信息,然後自動計算出結果。為了搞清楚計算過程,我們得到計算公式。梳理需求,得到下圖結果:根據上面的分析,試著補全代碼,將計算公式寫在下面,運行程序。