python @符号用法的简单理解
一、用作函数修饰符
作用是为现有函数增加额外的功能,常用于插入日志、性能测试、事务处理等等
创建函数修饰符的规则:
(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 decorater(*args, **kwargs): 4 func(*args, **kwargs) 5 print(type + '*'*10 + func.__name__) 6 return decorater 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
既可以通过类调用,也可以通过类的实例对象来调用