網站首頁 Python教程 正文
前言
本文對 Python 中的函數式編程技術進行了簡單的入門介紹。
在 Python 中,函數是「頭等公民」(first-class)。也就是說,函數與其他數據類型(如 int)處于平等地位。
因而,我們可以將函數賦值給變量,也可以將其作為參數傳入其他函數,將它們存儲在其他數據結構(如 dicts)中,并將它們作為其他函數的返回值。
把函數作為對象
由于其他數據類型(如 string、list 和 int)都是對象,那么函數也是 Python 中的對象。我們來看示例函數 foo,它將自己的名稱打印出來:
def?foo():
???print("foo")
由于函數是對象,因此我們可以將函數 foo 賦值給任意變量,然后調用該變量。
例如,我們可以將函數賦值給變量 bar:
bar?=?foo
bar()
#will?print?"foo"?to?the?console
語句 bar = foo 將函數 foo 引用的對象賦值給變量 bar。
把對象作為函數
當對象可調用時(callable),它們與函數一樣,如 object()。這是通過?call?方法實現的。
示例如下:
class?Greeter:
???def?__init__(self,?greeting):
??????self.greeting?=?greeting
???def?__call__(self,?name):
??????return?self.greeting?+?"?"?+?name
每一次配置 Greeter 類的對象時,我們都會創建一個新的對象,即打招呼時可以喊的新名字。
如下所示:
morning?=?Greeter("good?morning")?#creates?the?callable?object
morning("john")?#?calling?the?object
#prints?"good?morning?john"?to?the?console
我們可以調用 morning 對象的原因在于,我們已經在類定義中使用了?call?方法。為了檢查對象是否可調用,我們使用內置函數 callable:
callable(morning)?#true
callable(145)?#false.?int?is?not?callable.?
數據結構內的函數
函數和其他對象一樣,可以存儲在數據結構內部。例如,我們可以創建 int to func 的字典。當 int 是待執行步驟的簡寫時,這就會派上用場。
#?store?in?dictionary
mapping?=?{
???0?:?foo,
???1?:?bar
}
x?=?input()?#get?integer?value?from?user
mapping[x]()?#call?the?func?returned?by?dictionary?access
類似地,函數也可以存儲在多種其他數據結構中。
把函數作為參數和返回值
函數還可以作為其他函數的參數和返回值。接受函數作為輸入或返回函數的函數叫做高階函數,它是函數式編程的重要組成部分。
高階函數具備強大的能力。就像《Eloquent JavaScript》中解釋的那樣:
「高階函數允許我們對動作執行抽象,而不只是抽象數值。」
我們來看一個例子。假設我們想對一個項目列表(list of items)執行迭代,并將其順序打印出來。我們可以輕松構建一個 iterate 函數:
def?iterate(list_of_items):
????for?item?in?list_of_items:
????????print(item)
看起來很酷吧,但這只不過是一級抽象而已。如果我們想在對列表執行迭代時進行打印以外的其他操作要怎么做呢?
這就是高階函數存在的意義。我們可以創建函數 iterate_custom,待執行迭代的列表和要對每個項應用的函數都是 iterate_custom 函數的輸入:
def?iterate_custom(list_of_items,?custom_func):
???for?item?in?list_of_items:
????????custom_func(item)
這看起來微不足道,但其實非常強大。
我們已經把抽象的級別提高了一層,使代碼具備更強的可重用性。現在,我們不僅可以在打印列表時調用該函數,還可以對涉及序列迭代的列表執行任意操作。
函數還能被返回,從而使事情變得更加簡單。就像我們在 dict 中存儲函數一樣,我們還可以將函數作為控制語句,來決定適合的函數。
例如:
def?add(x,?y):
????return?x?+?y
def?sub(x,?y):
????return?x?-?y
def?mult(x,?y):
????return?x?*?y
def?calculator(opcode):
????if?opcode?==?1:
???????return?add
????elif?opcode?==?2:
???????return?sub
????else:
???????return?mult?
my_calc?=?calculator(2)?#my?calc?is?a?subtractor
my_calc(5,?4)?#returns?5?-?4?=?1?
my_calc?=?calculator(9)?#my?calc?is?now?a?multiplier
my_calc(5,?4)?#returns?5?x?4?=?20.?
嵌套函數
函數還可以在其他函數內部,這就是「內部函數」。內部函數在創建輔助函數時非常有用,輔助函數即作為子模塊來支持主函數的小型可重用函數。
在問題需要特定函數定義(參數類型或順序)時,我們可以使用輔助函數。這種不遵循傳統做法的操作使得解決問題變得更加簡單。
假設你想定義一個斐波那契函數 fib(n),該函數只有一個參數 n,我們必須返回第 n 個斐波那契數。
定義此類函數的一種可行方式是:使用輔助函數來追蹤斐波那契數列的前兩個項(因為斐波那契數是前兩個數之和)。
def?fib(n):
????def?fib_helper(fk1,?fk,?k):
????????if?n?==?k:
???????????return?fk
????????else:
???????????return?fib_helper(fk,?fk1+fk,?k+1)
????if?n?<=?1:
???????return?n
????else:
???????return?fib_helper(0,?1,?1)
將該計算從函數主體移到函數參數,這具備非常強大的力量。因為它減少了遞歸方法中可能出現的冗余計算。
單表達式函數(Lambda 表達式)
如果我們想在未給函數命名之前寫一個函數要怎么做?如果我們想寫一個簡短的單行函數(如上述示例中的函數 foo 或 mult)要怎么做?
我們可以在 Python 中使用 lambda 關鍵字來定義此類函數。
示例如下:
mult?=?lambda?x,?y:?x?*?y
mult(1,?2)?#returns?2
該 mult 函數的行為與使用傳統 def 關鍵字定義函數的行為相同。
注意:lambda 函數必須為單行,且不能包含程序員寫的返回語句。
事實上,它們通常具備隱式的返回語句(在上面的示例中,函數想表達 return x * y,不過我們省略了 lambda 函數中的顯式返回語句)。
lambda 函數更加強大和精準,因為我們還可以構建匿名函數(即沒有名稱的函數):
(lambda?x,?y:?x?*?y)(9,?10)?#returns?90
當我們只需要一次性使用某函數時,這種方法非常方便。例如,當我們想填充字典時:
import?collections
pre_fill?=?collections.defaultdict(lambda:?(0,?0))
#all?dictionary?keys?and?values?are?set?to?0
接下來我們來看 Map、Filter 和 Reduce,以更多地了解 lambda。
Map、Filter 和 Reduce
Map
map 函數基于指定過程(函數)將輸入集轉換為另一個集合。這類似于上文提到的 iterate_custom 函數。
例如:
def?multiply_by_four(x):
????return?x?*?4
scores?=?[3,?6,?8,?3,?5,?7]
modified_scores?=?list(map(multiply_by_four,?scores))
#modified?scores?is?now?[12,?24,?32,?12,?20,?28]
在 Python 3 中,map 函數返回的 map 對象可被類型轉換為 list,以方便使用。現在,我們無需顯式地定義 multiply_by_four 函數,而是定義 lambda 表達式:
modified_scores?=?list(map(lambda?x:?4?*?x,?scores))
當我們想對集合內的所有值執行某項操作時,map 函數很有用。
Filter
就像名稱所顯示的那樣,filter 函數可以幫助篩除不想要的項。例如,我們想要去除 scores 中的奇數,那么我們可以使用 filter:
even_scores?=?list(filter(lambda?x:?True?if?(x?%?2?==?0)?else?False,?scores))
#even_scores?=?[6,?8]
由于提供給 filter 的函數是逐個決定是否接受每一個項的,因此該函數必須返回 bool 值,且該函數必須是一元函數(即只使用一個輸入參數)。
Reduce
reduce 函數用于「總結」或「概述」數據集。例如,如果我們想要計算所有分數的總和,就可以使用 reduce:
sum_scores?=?reduce((lambda?x,?y:?x?+?y),?scores)
#sum_scores?=?32
這要比寫循環語句簡單多了。注意:提供給 reduce 的函數需要兩個參數:一個表示正在接受檢查的項,另一個表示所用運算的累積結果。
原文鏈接:https://juejin.cn/post/7100001304256184350
- 上一篇:Flask虛擬環境配置
- 下一篇:Python?自動備份腳本的示例代碼_python
相關推薦
- 2022-04-12 常見的狀態碼出現原因200、301、302、403、404、500、503
- 2022-05-11 C#實現搶紅包算法的示例代碼_C#教程
- 2022-11-23 解決vant組件van-list 首屏加載兩次的情況
- 2022-12-15 Apache中偽靜態Rewrite的使用方法和URL重寫規則表達式講解_Linux
- 2023-05-14 Python中tkinter無法同時顯示多個image的解決方法及pack與place解析_pyth
- 2022-10-05 python?slack桌面自動化開發工具_python
- 2022-04-25 ASP.NET?Core?MVC中過濾器工作原理介紹_實用技巧
- 2023-03-27 Python中反轉二維數組的行和列問題_python
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支