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

既可以通过类调用,也可以通过类的实例对象来调用