一 生成器
1什么是生產(chǎn)器
在python中芋类,一邊循環(huán)一邊計算的機制,稱為生成器(generator)劣光。
如果直接創(chuàng)建一個列表驶乾,而這個列表內(nèi)元素量居多,這時不僅占用很大的儲存空間叉信,如果只需要里面的前幾個元素時,那么大多數(shù)元素所占空間都浪費了。
如果可以通過算法推算出列表中的元素屡贺,可以通過循壞來計算得到需要的元素,不需要創(chuàng)建完整的list锌杀,節(jié)省空間甩栈。
2 創(chuàng)建生成器
①將列表生成式的[ ]改成()
生成器保存的是算法,每次調(diào)用next(f)糕再,就計算f下一個元素的值量没,直到計算到最后一個元素沒有元素時,拋出StopIteration異常突想。
但不斷用next()不方便殴蹄,一般是用for循環(huán),因為生成器也是可迭代對象
3創(chuàng)建生成器方法②
生成器非常強大猾担,如果算法比較復雜無法用for循環(huán)來實現(xiàn)的話袭灯,還可以用函數(shù)來實現(xiàn)。
比如绑嘹,著名的斐波拉契數(shù)列(Fibonacci):
1, 1, 2, 3, 5, 8, 13, 21, 34, ...
fib函數(shù)實際上是定義了斐波拉契數(shù)列的推算規(guī)則稽荧,可以從第一個元素開始,推算出后續(xù)任意的元素工腋,這種邏輯其實非常類似generator姨丈。
print(b)改為yield b,函數(shù)就可以變成生成器了
遇到y(tǒng)ield函數(shù)會停止夷蚊,在循環(huán)過程中不斷調(diào)用yield就不會中斷构挤。
但是用for循環(huán)調(diào)用generator時,發(fā)現(xiàn)拿不到generator的return語句的返回值惕鼓。如果想要拿到返回值筋现,必須捕獲StopIteration錯誤,返回值包含在StopIteration的value中:
4send
def fun():
? ? ? ? ?i=0
? ? ? ? ?while i<5:
? ? ? ? ? ? ? ? temp=yield i
? ? ? ? ? ? ? ? print(temp)
? ? ? ? ? ? ? ? ?i+=1
執(zhí)行到y(tǒng)ield時,fun函數(shù)暫時保存矾飞,返回i的值一膨,temp接收下次c.send('')發(fā)送過來的值,c.next()等價c.send(None)
5 多任務(wù)
多任務(wù)實現(xiàn)方式洒沦,協(xié)程
6 總結(jié)
生成器是這樣一個函數(shù)豹绪,它記住上一次返回時在函數(shù)體中的位置。對生成器函數(shù)的第二次(或第n次)調(diào)用跳轉(zhuǎn)至該函數(shù)中間申眼,而上次調(diào)用的所有局部變量都保持不變瞒津。
生成器不僅“記住”了它數(shù)據(jù)狀態(tài);生成器還“記住”了它在流控制構(gòu)造(在命令式編程中括尸,這種構(gòu)造不只是數(shù)據(jù)值)中的位置巷蚪。
生成器的特點:
1.節(jié)約內(nèi)存
2.迭代到下一次的調(diào)用時,所使用的參數(shù)都是第一次所保留下的濒翻,即是說屁柏,在整個所有函數(shù)調(diào)用的參數(shù)都是第一次所調(diào)用時保留的,而不是新創(chuàng)建的
二 迭代器
? ? ? ? 迭代是訪問集合元素的一種方式有送,迭代器是可以記住遍歷的位置的對象淌喻,迭代器對象從幾何的第一元素開始訪問,直到所有元素被訪問完結(jié)束雀摘,迭代器智能往前不會后退裸删。
1可迭代對象
可以用于for循環(huán)的數(shù)據(jù)類型:
一類是集合數(shù)據(jù)類型:list,tuple,dict,set,str等
一類是generator届宠,包括生成器和帶yield的genera function
可以直接用于for循環(huán)的對象成為可迭代對象:Iterable
2判斷是否可以迭代
用isinstance()判斷一個對象是否是Iterable對象
生成器不但可以作用于for循環(huán)烁落,還可以被next()函數(shù)不斷調(diào)用并返回下一個值乘粒,直到最后拋出StopIteration錯誤表示無法繼續(xù)返回下一個值了
3判斷對象是否為一個迭代器
可以被next()函數(shù)不斷返回下一個值得對象成為迭代器:Iterator
由此可見豌注,生成器一定是迭代器
可以使用instance()判斷一個對象是否是Iterator
4 iter()函數(shù)
生成器都是Iterator對象,但list灯萍,dict轧铁,str雖然是Iterable但不是Iterator
把list,dict,str等變成Iterator 可以使用iter()函數(shù):
5總結(jié)
凡是可以用for循環(huán)的對象都是Iterable類型
凡是可以作用于next()函數(shù)的對象都是Iterable類型
集合數(shù)據(jù)類型如list,dict,str是Iterable但不是Iterator,不過可以通過iter()函數(shù)獲得一個Iterator對象
目的:使用集合時減少占用內(nèi)存
三 閉包
1函數(shù)引用
2 什么是閉包
在函數(shù)內(nèi)部再定義一個函數(shù)旦棉,并且這個函數(shù)用到了外邊函數(shù)的變量齿风,那么將這個函數(shù)以及用到的一些變量稱之為閉包。
3 閉包的實例
在下面的例子中绑洛,函數(shù)line與變量a,b構(gòu)成閉包救斑,在創(chuàng)建閉包的時候,我們通過line_conf的參數(shù)a,b說明了這兩個變量的取值真屯,這樣脸候,我們就確定了函數(shù)的最終形式(y = x + 1和y = 4x + 5)。我們只需要變換參數(shù)a,b,就可以獲得不同的直線表達函數(shù)运沦。由此泵额,我們可以看到,閉包也具有提高代碼可復用性的作用携添。
4閉包的特點
1.閉包似優(yōu)化了變量嫁盲,原來需要類對象完成的工作,閉包也可以完成
2.由于閉包引用了外部函數(shù)的局部變量烈掠,則外部函數(shù)的局部變量沒有及時釋放羞秤,消耗內(nèi)存