你們好,我又回來了,今天我想和你們分享的是Python很是重要的幾個內置函數:map,filter,reduce, zip。
它們都是處理序列的便捷函數。這很大程度上歸功於函數式編程的世界。咱們能夠利用它們把一些小函數應用於一個序列的全部元素。從而節省編寫顯式循環的時間。python
另外,這些中的每個都是純函數,有返回值。所以咱們能夠容易地將函數的返回結果用表達式來表示。編程
好了,又到了大白話時間,爲何用它們,就是能夠簡化咱們的代碼,更簡潔高效的執行一些須要用到循環迭代爲主的任務,接下來讓咱們一個個來看數組
函數構造
map()函數的主要做用是能夠把一個方法依次執行在一個可迭代的序列上,好比List等,具體的信息以下:app
好了,大白話就是利用map咱們能夠把一個函數fun 執行到序列iter的每個元素上,用例子很是好理解~編程語言
基礎用法:
下面先讓咱們看一個小例子,假設如今咱們有一個List,包含1~5五個數字,咱們想要讓每個數+1,若是不知道map這個函數以前,咱們的解決方案是這樣的:函數式編程
numbers = [1, 2, 3, 4, 5] for i in range(0,len(numbers)): #對每一個元素加1 numbers[i]+=1 print(numbers) Out:[2, 3, 4, 5, 6]
或者是這樣的:函數
numbers = [1, 2, 3, 4, 5] result = [] for n in numbers: result.append(n+1) print(result) Out:[2, 3, 4, 5, 6]
可是顯然,不管怎麼作都會涉及到寫循環,這裏就是map函數的用武之地了,咱們能夠用map函數這樣實現:學習
def add_one(n): return n + 1 numbers = [1, 2, 3, 4, 5] result = map(add_one, numbers) print(result) print(type(result)) print(list(result)) Out:<map object at 0x00000260F508BE80> <class 'map'> [2, 3, 4, 5, 6]
這裏想必聰明的你發現了map的好處,在優化精簡代碼的同時,某種程度上講實現了方法和循環部分的分離,這裏咱們能夠發現map返回就是map類,咱們這裏傳遞的序列是List,最後輸出時通過類型轉換也是list測試
在傳遞序列時只要這個序列是可迭代的就好,不必定非要List,好比咱們換一種:優化
def add_one(n): return n + 1 numbers = (1, 2, 3, 4, 5) #序列爲元組 result = map(add_one, numbers) print(tuple(result)) # Out:(2, 3, 4, 5, 6)
輸入的序列爲一樣能夠迭代的元組,輸出時咱們也選擇元組,效果同樣的。
更進一步
還用剛纔的例子,爲了更加簡潔,咱們能夠用lambda函數配合map使用,具體實現以下:
numbers = (1, 2, 3, 4, 5) # 迭代對象爲tuple result = map(lambda x: x + 1, numbers) print(list(result)) # 輸出對象爲list Out:[2, 3, 4, 5, 6]
更加簡潔優雅了對吧!!這個lambad函數我以後會說,今天它不是主角哈哈,先一帶而過。
讓咱們從新把目光轉移到map上來,除了剛纔的用法,還要一種狀況也十分常見,讓咱們看下面的例子:
# List of strings words = ['paris', 'xiaobai','love'] # 把數組中每一個元素變爲List test = list(map(list, words)) print(test) Out: [['p', 'a', 'r', 'i', 's'], ['x', 'i', 'a', 'o', 'b', 'a', 'i'], ['l', 'o', 'v', 'e']]
words是一個只包含字符串類型元素的list,咱們用map能夠實現將words的每個元素所有轉化爲list類型,這裏有一點必定要注意,能實現的前提必定是每一個元素都是能夠迭代的類型,若是出現瞭如int類型的元素,就會出錯啦:
# List of strings words = [18,'paris', 'xiaobai','love'] # 把數組中每一個元素變爲List test = list(map(list, words)) print(test) Out:TypeError: 'int' object is not iterable
你們一看錯誤類型相比馬上就明白啦,因此正確的使用方法必定是相似這種:
nums = [3,"23",-2] print(list(map(float,nums))) Out: [3.0, 23.0, -2.0]
總之就是類型要注意,今天我就拋磚引玉簡單介紹一下map,具體的用法你們能夠自行開發哈,我也在不斷學習中
函數構造
filter()方法藉助於一個函數來過濾給定的序列,該函數測試序列中的每一個元素是否爲真。
簡而言之就是filter能夠幫助咱們根據給出的條件過濾一組數據並返回結果
基礎用法:
讓咱們先看一個例子:
# 過濾元音的方法 def fun(variable): letters = ['a', 'e', 'i', 'o', 'u'] if (variable in letters): return True else: return False # 給定序列 sequence = ['I', 'l', 'o', 'v', 'e', 'p', 'y','t','h','o','n'] # 根據條件得出結果 filtered = list(filter(fun, sequence)) print(filtered) Out:['o', 'e', 'o']
這裏咱們建立一個能夠提取元音字母的方法fun,給定的可迭代序列爲list,以後就能夠用filter方法很容易的提取出結果啦,再看一個相似例子:
# 判斷爲正數 def positive(num): if num>0: return True else: return False #判斷偶數 def even(num): if num % 2==0: return True else: return False numbers=[1,-3,5,-20,0,9,12] positive_nums = list(filter(positive, numbers)) print(positive_nums) # 輸出正數 list even_nums = tuple(filter(even,numbers)) print(even_nums) #輸出偶數 tuple Out:[1, 5, 9, 12] (-20, 0, 12)
看到這裏相比你們已經知道filter的基礎用法啦, 要先有一個,能返回True或者False的方法,或者表達式做爲過濾條件就行啦
更進一步
這裏其實和map同樣了,基本上最簡潔的用法都是和lambda混在一塊兒,好比下面咱們想要把剛纔的一大串代碼壓縮一下:
numbers = [0, 1, 2, -3, 5, -8, 13] # 提取奇數 result = filter(lambda x: x % 2, numbers) print("Odd Numbers are :",list(result)) # 提取偶數 result = filter(lambda x: x % 2 == 0, numbers) print("Even Numbers are :",list(result)) #提取正數 result = filter(lambda x: x>0, numbers) print("Positive Numbers are :",list(result)) Out:Odd Numbers are : [1, -3, 5, 13] Even Numbers are : [0, 2, -8] Positive Numbers are : [1, 2, 5, 13]
" 爽啊!爽死了!" 郭德綱看到後這麼評價,lambda我平時用的很少,可是寫到這裏,我也以爲要好好學習它了,畢竟和其餘編程語言相比,可能這中用法纔是python提倡的理念之一:高效簡潔,
函數構造
Reduce是一個很是有用的函數,用於在列表上執行某些計算並返回結果。它將滾動計算應用於列表中的連續值。例如,若是要計算整數列表的累積乘,或者求和等等
在Python 2中,reduce()是一個內置函數。可是,在Python 3中,它被移動到functools模塊。所以,要使用前咱們須要導入,這裏個人環境是Python 3.6
基礎用法:
先看一個求累加和的小栗子:
from functools import reduce # Python 3 def do_sum(x1, x2): return x1 + x2 print(reduce(do_sum, [1, 2, 3, 4])) Out:10
再看一個累積乘法的例子:
from functools import reduce # Python 3 def multiply(x, y): return x*y numbers = [1,2,3,4] print(reduce(multiply, numbers)) Out:24
更進一步:
仍是和lambda混搭,更加簡潔:
from functools import reduce # Python 3 numbers = [1,2,3,4] result_multiply = reduce((lambda x, y: x * y), numbers) result_add = reduce((lambda x,y: x+y), numbers) print(result_multiply) print(result_add) Out:24 10
函數構造
zip()的目的是映射多個容器的類似索引,以便它們能夠僅做爲單個實體使用。
基礎用法:
其實以前咱們在講dict的建立方法時提到過它,這裏重新回顧一下:
keys = ['name','age'] values = ['xiaobai',18] my_dict = dict(zip(keys,values)) print(my_dict) Out:{'name': 'xiaobai', 'age': 18}
zip能夠支持多個對象,好比下面的例子
name = [ "xiaobai", "john", "mike", "alpha" ] age = [ 4, 1, 3, 2 ] marks = [ 40, 50, 60, 70 ] # using zip() to map values mapped = list(zip(name, age, marks)) print ("The zipped result is : "mapped) Out:The zipped result is : [('xiaobai', 4, 40), ('john', 1, 50), ('mike', 3, 60), ('alpha', 2, 70)]
這裏咱們能夠很容易的的把name,age,marks這三個list裏面相同index的值映射打包在一塊兒
更進一步:
經過上面的例子,咱們發現能夠很容易的以相似1對1的形式把不一樣對象的同一索引位置的值打包起來,那若是是解包呢?也是相似的,就是多了一個 * 而已
names, ages, marks = zip(*mapped) print ("The name list is : ",names) print ("The age list is : ",ages) print ("The marks list is : ",marks) Out: The name list is : ('xiaobai', 'john', 'mike', 'alpha') The age list is : (4, 1, 3, 2) The marks list is : (40, 50, 60, 70)
今天主要爲你們介紹了map,filter,reduce,zip四個高效的python內置函數的用法,我也是剛剛接觸,瞭解不夠深刻,若是介紹的有錯誤或者歧義還請你們多多諒解和包容,若是有大神能夠進一步補充說明必定要寫個評論呀,讓咱們一塊兒進步。
最後爲你們講個悲傷的故事: