Fork me on GitHub

python装饰器的深度探究

1.讲装饰器一般讲到这种代码就可以了,但这篇博客会介绍更多:

def deco(func):
    def wrapper():
        print("start")
        func() #调用函数
        print("end")
    return wrapper
 
@deco
def myfun():
    print "run"
 
myfun()

2.装饰任意参数的函数:

def deco(func):
    def warpper(*args,**kwargs):
        print("start")
        func(*args,**kwargs)
        print("end")
    return warpper
     
@deco
def myfun1(param1):
    print "run with param %s"%(param1)

装饰器会重写函数的名字和注释文档,@wraps(func)可以解决这个问题

from functools import wraps

def deco(func):
    @wraps(func)
    def warpper(*args,**kwargs):
        print("start")
        func(*args,**kwargs)
        print("end")
    return warpper

3.django自定义装饰器实现登录验证

def Check_Login(func):  #自定义登录验证装饰器
    def warpper(request,*args,**kwargs):
        is_login = request.session.get('IS_LOGIN', False)
        if is_login:
            func(request,*args,**kwargs)
        else:
            return HttpResponseRedirect("/polls/login_user")
    return warpper
 
def login_user(request):
    if request.method == 'POST':
        form = LoginForm(request.POST)
        if form.is_valid():
		    # 获取post数据,例如 {'username': u'yang1', 'password': 111}
            all_data = form.clean()   
			u = all_data['Form_username']
			p = all_data['Form_password']
            exist = User.objects.filter(username = u,password = p).first()
            if exist:
                request.session['IS_LOGIN'] = True  #设置session的随机字段值
                request.session['uname'] = exist.username   #设置uname字段为登录用户
                return HttpResponseRedirect('/polls/home')
            else:
                return HttpResponse("账户或密码错误")
    else:
        form = LoginForm()
    return render(request, 'polls/login_user.html', {'form': form})
 
@Check_Login
def home(request):
        username = request.session.get('uname', False)   #获取登录用户名
        return render(request, 'polls/home.html', {'username': username})

4.带参数的装饰器

装饰器接受一个函数作为参数,这个毋庸置疑.但是有时候我们需要装饰器接受另外的参数,

此时需要再加一层函数,实际上是定义了一个生成装饰器的工厂函数,

调用它,搭配需要的参数,来返回合适的装饰器.

from functools import wraps
 
def logit(logfile='out.log'):
    def logging_decorator(func):
        @wraps(func)
        def wrapped_function(*args, **kwargs):
            log_string = func.__name__ + " was called"
            print(log_string)
            # 打开logfile,并写入内容
            with open(logfile, 'a') as opened_file:
                # 现在将日志打到指定的logfile
                opened_file.write(log_string + '\n')
            return func(*args, **kwargs)
        return wrapped_function
    return logging_decorator
 
@logit()
def myfunc1():
    pass

5.装饰器类

比方说有时你只想打日志到一个文件;

而有时你想把引起你注意的问题发送到一个email,同时也保留日志,留个记录.

from functools import wraps
 
class logit(object):
    def __init__(self, logfile='out.log'):
        self.logfile = logfile
 
    def __call__(self, func):
        @wraps(func)
        def wrapped_function(*args, **kwargs):
            log_string = func.__name__ + " was called"
            print(log_string)
            # 打开logfile并写入
            with open(self.logfile, 'a') as opened_file:
                # 现在将日志打到指定的文件
                opened_file.write(log_string + '\n')
            # 现在,发送一个通知
            self.notify()
            return func(*args, **kwargs)
        return wrapped_function
 
    def notify(self):
        # logit只打日志,不做别的
        pass

使用类装饰器主要依靠类的__call__方法,当使用 @ 形式将装饰器附加到函数上时,

就会调用此方法.这个实现有一个附加优势,在于比嵌套函数的方式更加整洁.

给 logit 创建子类,来添加 email 的功能:

class email_logit(logit):
    '''
    一个logit的实现版本,可以在函数调用时发送email给管理员
    '''
    def __init__(self, email='admin@myproject.com', *args, **kwargs):
        self.email = email
        super(email_logit, self).__init__(*args, **kwargs)
 
    def notify(self):
        # 发送一封email到self.email
        # 这里就不做实现了
        pass

从现在起,@email_logit将会和@logit产生同样的效果,

但是在打日志的基础上,还会多发送一封邮件给管理员.

6.装饰器的执行顺序总结

def decorator_a(func):
    print('Get in decorator_a')

    def inner_a(*args, **kwargs):
        print('Get in inner_a')
        return func(*args, **kwargs)
    return inner_a


def decorator_b(func):
    print('Get in decorator_b')

    def inner_b(*args, **kwargs):
        print('Get in inner_b')
        return func(*args, **kwargs)
    return inner_b


@decorator_a
@decorator_b
def f(x):
    print('Get in f')
    return x * 2
当你不调用就这么执行时,返回结果是:
Get in decorator_b
Get in decorator_a
当你调用了f(1),返回结果是:
Get in decorator_b
Get in decorator_a
Get in inner_a
Get in inner_b
Get in f

所以:定义过程是自下向上,执行顺序是自上向下.

还有一个<装饰器带类参数>,有点深,看不懂,用到的时候再研究.

参考地址:http://www.runoob.com/w3cnote/python-func-decorators.html

参考地址:https://www.cnblogs.com/honey-badger/p/8978168.html

posted @ 2018-11-16 16:35  法外狂徒  阅读(224)  评论(0编辑  收藏  举报