首頁 > 軟體

python中的迭代器,生成器與裝飾器詳解

2022-02-18 13:03:36

迭代器

每一個可迭代類內部都要實現__iter__()方法,返回一個迭代類物件,迭代類物件則定義了這個可迭代類如何迭代。

for迴圈呼叫list本質上是是呼叫了list的迭代器進行迭代。

# 對list進行for迴圈本質上是呼叫了list的迭代器
list = [1,2,3,4]

# for 迴圈呼叫
for elem in list:
    print(elem)

# 迭代器呼叫
list_iter = list.__iter__()
while True:
   try:
    print(next(list_iter)) 
   except StopIteration:
       break

實現一個自己自定的迭代類,規定迭代一個可迭代的資料結構為“倒計時”模式。

# 可迭代物件類
class CountDown(object):
    def __init__(self,num):
        self.num = num
    def __iter__(self):
        return MyIterator(self.num)

# 迭代類
class MyIterator(object):
    def __init__(self,num):
        self.NUM= num
        self.FINAL = 0
        self.now = num
        
    def __iter__(self):
        return self
    
    def __next__(self):
        step = 1 if self.NUM<self.FINAL else -1
        while self.now != self.FINAL:
            self.now += step
            return self.now - step
        raise StopIteration
        
cd_pos = CountDown(5)
cd_neg = CountDown(-5)
for i,j in zip(cd_pos,cd_neg):
    print(f'pos:{i}tneg:{j}')

生成器

含有yield指令的函數可以稱為生成器,它可以將函數執行物件轉化為可迭代的物件。這樣就可以像debug一樣一步一步推進函數。可以實現的功能是可以實現讓函數內部暫停,實現了程式的非同步功能,這樣可以及進行該函數與外部構件的資訊互動,實現了系統的解耦。

from  collections import Iterable
def f():
    pass
# 含有yield指令的函數可以稱為生成器
def g():
    yield()
print(type(f()),isinstance(f(),Iterable))
print(type(g()),isinstance(g(),Iterable))

使用生成器可以降低系統的耦合性

import os
# 生成器是迭代器的一種,讓函數物件內部進行迭代
# 可以實現讓函數內部暫停,實現了程式的非同步功能,同時也實現瞭解耦。

def my_input():
    global str    
    str = input('input a line')
    pass

def my_write():
    with open('workfile.txt','w') as f:
        while(str):
            f.write(str+'n')
            yield()
        return 


mw = my_write()
while(True):
    my_input()
    try:
        next(mw)
    except StopIteration:
        pass
    if not str:
        break   

裝飾器

裝飾器封裝一個函數,並且用這樣或者那樣的方式來修改它的行為。

不帶引數的裝飾器

# 不帶引數的裝飾器
from functools import wraps
# 裝飾器封裝一個函數,並且用這樣或者那樣的方式來修改它的行為。
def mydecorator(a_func):
    @wraps(a_func)  #宣告這個註解就可以不重寫傳入的函數,只是呼叫的時候wrap一下。不加的話,a_func函數可以看作被重寫為wrapTheFunction.
    def wrapTheFunction():
        print(f"function in {id(a_func)} starts...")
        a_func()
        print(f"function in {id(a_func)} ends...")
    return wrapTheFunction

# 在函數定義前加入此註解就可以將函數傳入裝飾器幷包裝
@mydecorator
def f():
    print('hi')
    pass

f()
print(f.__name__)

帶引數的裝飾器(實現輸出到自定義的紀錄檔檔案)

# 帶引數的裝飾器(實現輸出到自定義的紀錄檔檔案)
from functools import wraps

def logit(logfile='out.log'):
    def mydecorator2(a_func):
        @wraps(a_func)
        def wrapTheFunction(*args, **kwargs):   # 這個保證了函數可以含有任意形參
            log_string = a_func.__name__ + " was called"
            print(log_string)
            # 開啟logfile,並寫入內容
            with open(logfile, 'a') as opened_file:
                # 現在將紀錄檔打到指定的logfile
                opened_file.write(log_string + 'n')
                return a_func(*args, **kwargs)
        return wrapTheFunction
    return mydecorator2

# func group1 
@ logit('out1.log')
def func1(str):
    print(str)
    pass
@ logit('out2.log')
def func2(): pass
            
func1('I have a foul smell')
func2()        

實現一個裝飾器類(這樣寫可以簡化裝飾器函數,並且提高封裝性)

# 帶引數的裝飾器(實現輸出到自定義的紀錄檔檔案)
from functools import wraps

def logit(logfile='out.log'):
    def mydecorator2(a_func):
        @wraps(a_func)
        def wrapTheFunction(*args, **kwargs):   # 這個保證了函數可以含有任意形參
            log_string = a_func.__name__ + " was called"
            print(log_string)
            # 開啟logfile,並寫入內容
            with open(logfile, 'a') as opened_file:
                # 現在將紀錄檔打到指定的logfile
                opened_file.write(log_string + 'n')
                return a_func(*args, **kwargs)
        return wrapTheFunction
    return mydecorator2

# func group1 
@ logit('out1.log')
def func1(str):
    print(str)
    pass
@ logit('out2.log')
def func2(): pass
            
func1('I have a foul smell')
func2()        
    

總結

到此這篇關於python中的迭代器,生成器與裝飾器詳解的文章就介紹到這了,更多相關python迭代器,生成器與裝飾器內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com