一、用作函數修飾符
作用是為現有函數增加額外的功能,常用於插入日志、性能測試、事務處理等等
創建函數修飾符的規則:
(1)修飾符是一個函數
(2)修飾符取被修飾函數為參數
(3)修飾符返回值取代被修飾函數
舉個例子:
1 def log(func): 2 def wrapper(): 3 print('...log begin...') 4 func() 5 print('...log end...') 6 return wrapper 7 8 @log 9 def test(): 10 print('test...') 11 12 test()
運行過程及結果:
log函數叫作修飾符或裝飾器,test是被修飾函數,裝飾器必須在前面定義
首先,test函數作為log函數的參數,執行log函數,然后log函數執行結果返回值為wrapper函數,又賦值給test,所以此時test函數實際上是wrapper函數。
故12行的代碼test()實際上是執行的wrapper(),即結果為:
...log begin...
test...
...log end...
再舉個例子:
1 def log(func): 2 def wrapper(*args,**kwargs): 3 print('...log begin...') 4 func(*args,**kwargs) 5 print('...log end...') 6 return wrapper 7 8 @log 9 def test1(arg1): 10 print('test1...',arg1) 11 return arg1 12 13 @log 14 def test2(arg1,arg2): 15 print('test2...',arg1,arg2) 16 return arg1+arg2 17 18 test1(a) 19 test2(a,bc)
此例子只是被修飾函數帶參數,運行過程同上一個例子,結果如下:
...log begin...
test1... a
...log end...
...log begin...
test2... a bc
...log end...
再來一個裝飾器帶參數的例子:
1 def log(type): 2 def wrapper(func): 3 def decoter(*args, **kwargs): 4 func(*args, **kwargs) 5 print(type + '*'*10 + func.__name__) 6 return decoter 7 return wrapper 8 9 @log('Debug') 10 def func_a(): 11 print('example') 12 13 if __name__ == '__main__': 14 func_a()
運行結果:
D:\Python-Project\venv\Scripts\python.exe D:/work/scripts/tickets/test.py example Debug**********func_a Process finished with exit code 0
二、用作類方法及靜態方法
Python Class類中有很多屬性及方法,其中方法主要有三種:實例方法、類方法和靜態方法
還是舉個例子:
#!/usr/bin/ env python # -*- coding:utf-8 -*- class Student(object): #類屬性 name = 'fdzwdt' def __init__(self,name): #實例對象屬性 self.name = name #實例方法,只能由實例對象來調用 def get_name_ins(self): return self.name #類方法,可以通過類來調用,也可以通過實例對象調用 @classmethod def get_name_cls(cls): return cls.name #靜態方法,可以通過類來調用,也可以通過實例對象調用 #主要功能就是對__init__構造函數進行重載 @staticmethod def get_name_sta(name): return Student(name).name st = Student('weidt') print('ins_name:'+st.get_name_ins()) print('cls_name:'+Student.get_name_cls()) print('sta_name:'+Student.get_name_sta('wdt'))
運行結果:
ins_name:weidt
cls_name:fdzwdt
sta_name:wdt
改變調用方式:
print('ins_name:'+Student.get_name_ins())
結果:
通過上例錯誤信息,可知實例方法只能通過實例對象調用
print('cls_name:'+st.get_name_cls()) print('sta_name:'+st.get_name_sta('wdt'))
結果:
cls_name:fdzwdt
sta_name:wdt
既可以通過類調用,也可以通過類的實例對象來調用