偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

程序員為何與函數(shù)式編程“墜入愛河”?

開發(fā) 前端
函數(shù)式編程發(fā)展至今已有60年的歷史,但是截至目前,它仍然算是比較小眾。盡管像Google這樣的大公司依賴于函數(shù)式編程的關(guān)鍵概念,但是普通程序員對此幾乎一無所知。

本文轉(zhuǎn)載自公眾號“讀芯術(shù)”(ID:AI_Discovery)。

函數(shù)式編程發(fā)展至今已有60年的歷史,但是截至目前,它仍然算是比較小眾。盡管像Google這樣的大公司依賴于函數(shù)式編程的關(guān)鍵概念,但是普通程序員對此幾乎一無所知。

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

[[340779]]

簡單來說,函數(shù)式編程就是為不可變變量構(gòu)建函數(shù)。與之相反,面向?qū)ο蟮木幊虅t是有一組相對固定的函數(shù),而用戶主要是修改或添加新變量。

由于函數(shù)式編程的特性,它非常適合完成諸如數(shù)據(jù)分析和機器學(xué)習(xí)之類的需求任務(wù)。但是這并不意味著用戶要告別面向?qū)ο蟮木幊蹋D(zhuǎn)而完全使用函數(shù)式編程。但用戶需要了解其基本原理,以便在適當?shù)臅r候使用它們以發(fā)揮優(yōu)勢。

一切都是為了消除副作用

要了解函數(shù)式編程,首先需要了解函數(shù)。函數(shù)是將輸入轉(zhuǎn)換為輸出的東西,它并不總是這么簡單。下面看一個Python中的函數(shù):

  1. def square(x): 
  2.     return x*x 

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

現(xiàn)在再思考這個函數(shù):

  1. lobal_list = []def append_to_list(x): 
  2.     global_list.append(x) 

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

請等一下!如果未事先定義global_list,那么該函數(shù)將不起作用,并且在經(jīng)過修改后仍輸出相同的列表。盡管global_list從未被視為函數(shù)的輸入,但使用函數(shù)時它也會發(fā)生改變:

  1. append_to_list(1) 
  2. append_to_list(2) 
  3. global_list 

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

[[340780]]

圖源:GitHub

不忠實于函數(shù)

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

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

有一個簡單的解決方法:忠于函數(shù)認定為輸入的內(nèi)容。

  1. newlist = []def append_to_list2(x, some_list): 
  2.    some_list.append(x)append_to_list2(1,newlist) 
  3. append_to_list2(2,newlist) 
  4. newlist 

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

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

函數(shù)式編程正在編寫純函數(shù)

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

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

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

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

[[340781]]

圖源:unsplash

函數(shù)式編程不僅只有Map和reduce

函數(shù)式編程中不包含循環(huán)結(jié)構(gòu)(Loops),請看下面這些Python中的循環(huán):

  1. integers = [1,2,3,4,5,6] 
  2. odd_ints = [] 
  3. squared_odds = [] 
  4. total = 0for i in integers: 
  5.     if i%2 ==1 
  6.         odd_ints.append(i)for i inodd_ints: 
  7.     squared_odds.append(i*i)for i insquared_odds: 
  8.     total += i 

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

  1. from functools import reduceintegers = [1,2,3,4,5,6] 
  2. odd_ints = filter(lambda n: n % 2 == 1, integers) 
  3. squared_odds = map(lambda n: n * n, odd_ints) 
  4. total = reduce(lambda acc, n: acc + n, squared_odds) 

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

  • Lambda functions:談及函數(shù)式編程的發(fā)展史時,許多人都會先提及l(fā)ambda函數(shù)的發(fā)明。盡管,lambda毫無疑問是函數(shù)式編程的基石,但這并不是根本原因。Lambda函數(shù)是可使程序發(fā)揮作用的工具。但是,lambda也可用于面向?qū)ο蟮木幊獭?/li>
  • Static typing:上面的示例不屬于靜態(tài)輸入,而是函數(shù)式的。即使靜態(tài)類型為代碼增加了一層額外的安全保護,但也并非一定要其函數(shù)化,不過這可能會是錦上添花。

一些語言對函數(shù)式編程更加友好

[[340782]]

圖源:unsplash

(1) Perl

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

(2) Java

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

(3) Scala

Scala是一個很有趣的語言:它的目標是統(tǒng)一面向?qū)ο蠛秃瘮?shù)式編程。很多人都覺得這很奇怪,因為函數(shù)式編程旨在徹底消除副作用,而面向?qū)ο蟮木幊虅t試圖將副作用保留在對象內(nèi)部。

話雖如此,許多開發(fā)人員將Scala視為一種可以幫助他們從面向?qū)ο缶幊踢^渡到函數(shù)式編程語言,這可能會幫助他們在未來幾年更容易完全過渡到函數(shù)式編程。

(4) Python

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

(5) Clojure

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

(6) Haskell

這是極少數(shù)純函數(shù)式和靜態(tài)類型的語言之一。盡管在開發(fā)過程中可能會耗費大量時間,但在調(diào)試程序時這些付出都會獲得巨大回報。它不像其他語言那樣容易學(xué)習(xí),但是絕對值得花時間學(xué)習(xí)。

程序員為何與函數(shù)式編程“墜入愛河”?

圖源:unsplash

與面向?qū)ο蟮木幊滔啾?,函?shù)式編程仍然小眾。但是,如果說在Python和其他語言中加入函數(shù)式編程原理意味著什么的話,那就是函數(shù)式編程正越來越受到關(guān)注。這完全說得通:函數(shù)式編程對于大型數(shù)據(jù)庫、并行編程和機器學(xué)習(xí)大有裨益。而在過去十年間,這些迎來了蓬勃發(fā)展。

雖然面向?qū)ο缶幊逃兄豢晒懒康膬?yōu)點,但函數(shù)代碼的優(yōu)點也不容忽視。只需要學(xué)習(xí)一些基本原理,就足以讓用戶成為一名開發(fā)人員,并為未來做好準備。

 

責(zé)任編輯:趙寧寧 來源: 今日頭條
相關(guān)推薦

2017-11-09 11:03:06

AMDIntelNVIDIA

2014-09-05 10:15:41

函數(shù)式編程

2013-11-14 10:05:25

程序員職業(yè)轉(zhuǎn)型

2013-01-17 09:54:12

程序員編程

2012-08-30 10:05:40

編程編程語言程序員

2013-07-15 13:45:16

程序員

2012-12-21 15:14:35

Java

2019-08-22 11:09:26

程序員技能開發(fā)者

2012-06-15 09:54:58

程序員編程開發(fā)

2014-06-13 11:03:19

程序員

2011-11-11 16:20:22

編程

2011-02-15 13:12:02

JavaC++

2017-06-08 14:25:46

Kotlin函數(shù)

2009-07-28 08:28:15

2014-11-10 09:46:57

程序員

2015-02-03 02:40:33

程序員盲人程序員

2010-03-22 09:32:42

自學(xué)編程

2012-10-22 14:17:42

函數(shù)式程序員

2023-09-07 09:46:44

程序員編程

2023-12-26 01:14:20

函數(shù)式編程死鎖
點贊
收藏

51CTO技術(shù)棧公眾號