写这篇随笔是因为今天自己在写插件和过滤方法的过程中碰壁了,折腾了好久终于稍微发现些问题,在此记下,以作备忘。

在看了xadmin的插件机制后,笔者也想使用该思想来扩展kadmin中视图的方法。

例如,在一个登陆视图中,一般的逻辑如下:

class LoginView(BaseAdminView):
'''登陆视图'''
auth_form=None#用于认证的表单类
login_template=None
title="" def update_login_params(self,defaults):
''' 用于在执行login视图函数前修改参数的钩子'''
return defaults
@never_cache
def get(self,request,*arg,**kwargs):
from django.contrib.auth.views import login
context=self.get_context(request,*arg,**kwargs)#获取父类的context
context.update({
'title':self.title,
'app_path':request.get_full_path,
REDIRECT_FIELD_NAME:request.get_full_path(),
})
defaults={
'extra_context':context,
'authentication_form':self.auth_form or AdminAuthenticationForm,
'template_name':self.login_template or 'kadmin/views/login.html',
}
       defaults=self.update_login_params(defaults) 

       return login(request,**defaults) @never_cache def post(self,request,*arg,**kwargs): return self.get(request,*arg,**kwargs)

我们希望能通过update_login_params方法,在执行login函数前更新context,就应该使用插件来接管该方法的执行以按需要进行一些修改。

这里有两种思路:

1、使用过滤器,过滤update_login_params的结果并返回

2、使用插件函数,控制update_login_params的执行(一般是活得update_login_params的执行结果 传入插件函数 由插件函数进行一些修改后再返回)

也可以用插件函数接管update_login_params,并自行决定是否执行update_login_params,或是在update_login_params执行前或后进行一些行为

从上述两点看,插件更为灵活和自由,控制范围更大,二过滤器只能就update_login_params的返回值进行操作并返回。

1、过滤器

定义过滤器装饰器@filter_hook

def filter_hook(func):
#过滤器装饰器
#用于对被hook的方法返回值进行再处理
func_name=func.__name__
@wraps(func)
def inner_func(self,*arg,**kwargs):
if getattr(self,'filters',None):
filter_infos=[(getattr(fi,'priority',10),fi) for fi in self.filters if fi.__name__==func_name]
filters=[fi for p,fi in sorted(filter_infos,key=lambda x:x[0],reverse=False)]
result=func(self,*arg,**kwargs)
for fi in filters:
result=fi(self,result,*arg,**kwargs)
return result
else:
return func(self,*arg,**kwargs)
return inner_func

源码说明:在视图对象self上遍历filters 得到过滤函数,过滤结果并返回结果

例如我们可以定义一个过滤器给say_hello的返回值加上括号

@set_attr(attr_name='__name__',value='say_hello')
def add_tag(self,result,o,*arg,**kwargs):
return "(%s)"%result
class Person:
name='akun'
sex="akun-male"
plugins=[MalePlugin(sex_num) for sex_num in range(4)]
filters=[add_tag] @filter_hook
@plugin_hook
def say_hello(self,go,*arg,**kwargs):
print("-----------done--------")
return "Hello i am %s"%self.name
if __name__=="__main__":
p=Person()
print(p.say_hello('go'))

说明:@set_attr是一个装饰器,负责设置函数的属性值,因为filter_hook在查找过滤函数时是找被hook的同名函数,所以需要把add_tag的__name__设为say_hello

2.插件

定义插件装饰器:

def plugin_chain(funcs,token,func,*arg,**kwargs):
if token==-1:
return func()
@wraps(func)
def _inner_func():
wrap_func=funcs[token]
arg_specs=getargspec(wrap_func)[0]
if len(arg_specs)==1:
func()
return wrap_func(*arg,**kwargs)
elif len(arg_specs)>=2 and arg_specs[1]=="__":
back=func
else:
back=func()
return wrap_func(back,*arg,**kwargs)
return plugin_chain(funcs,token-1,_inner_func,*arg,**kwargs) def plugin_hook(func):
func_name=func.__name__
@wraps(func)
def method(self,*arg,**kwargs):
@wraps(func)
def _inner_func():
return func(self,*arg,**kwargs)
if getattr(self,'plugins',None):
plugin_funcs=[(getattr(p,func_name),getattr(getattr(p,func_name),'priority',10))
for p in self.plugins if getattr(p,func_name,None) ]
print(plugin_funcs)
#对插件方法按照priority升序排列a
plugin_funcs=[p for p,priority in sorted(plugin_funcs,key=lambda x:x[1],reverse=False)]
return plugin_chain(plugin_funcs,len(plugin_funcs)-1,_inner_func,*arg,**kwargs)
else:
return func(self,*arg,**kwargs)
return method

源码说明:类似于@filter_hook只是,注意

        wrap_func=funcs[token]
arg_specs=getargspec(wrap_func)[0]
if len(arg_specs)==1:
func()
return wrap_func(*arg,**kwargs)
elif len(arg_specs)>=2 and arg_specs[1]=="__":
back=func
else:
back=func()
return wrap_func(back,*arg,**kwargs)

如果若用于接收被hook方法的返回值的参数名为"__"则,__会被设置为被hook的方法,也就是可以通过插件来接管被hook的方法。

我们定义一个插件来在执行say_hello前打印出被hook的方法

class BeforeHook:
@set_attr(attr_name='priority',value=6)
def say_hello(self,__,o,*arg,**kwargs):
print("the back is %s"%__)
return __(*arg,**kwargs)

这里的__会被设置为被hook的方法

注意:

    正是因为@plugin_hook是需要获取argspecs来得到位置参数的名称,用于判断插件函数是否接受被hook方法的返回值,以及是否在argsepcs[0]中是否

含有双下划线位置参数,如果有则把被hook的方法交给插件函数 而不是把结果交出去。所以基于以上两点,我们不能@filter_hook一个插件方法,因为那样会

包装一个新函数(参数列表为 self,*arg,**kwargs)给plugin_hook,那样的话plugin_hook得到的argsepcs[0]只有self,会认为插件方法不接受返回值,于是

直接调用

        if len(arg_specs)==1:
func()
return wrap_func(*arg,**kwargs)

故会产生错误

但是我们可以在被hook的方法上@filter_hook和@plugin_hook任意组合

插件和过滤器装饰器开发中的感悟-python-django的更多相关文章

  1. day20-Python运维开发基础(装饰器 / 类中的方法 / 类的方法变属性)

    1. 装饰器 / 类中的方法 / 类的方法变属性 # ### 装饰器 """ 定义:装饰器用于拓展原来函数功能的一种语法,返回新函数替换旧函数 优点:在不更改原函数代码的 ...

  2. 理解Python中的装饰器//这篇文章将python的装饰器来龙去脉说的很清楚,故转过来存档

    转自:http://www.cnblogs.com/rollenholt/archive/2012/05/02/2479833.html 这篇文章将python的装饰器来龙去脉说的很清楚,故转过来存档 ...

  3. python中闭包和装饰器的理解(关于python中闭包和装饰器解释最好的文章)

    转载:http://python.jobbole.com/81683/ 呵呵!作为一名教python的老师,我发现学生们基本上一开始很难搞定python的装饰器,也许因为装饰器确实很难懂.搞定装饰器需 ...

  4. Django模板自定义标签和过滤器,模板继承(extend),Django的模型层

    上回精彩回顾 视图函数: request对象 request.path 请求路径 request.GET GET请求数据 QueryDict {} request.POST POST请求数据 Quer ...

  5. 视频播放器开发中遇到的一些小问题MPMoviePlayerController

    1 开发环境是 xcode6  ipad3真机 ios8.1.1越狱 需要添加以下代码  ,否则真机测试没有外音,只有耳机   NSError *setCategoryError = nil;    ...

  6. Pytest的装饰器——parametrize中ids里包含中文,用例标题显示异常如何解决?

    在使用pytest做测试的过程中,经常会用到pytest.mark.parametrize来对批量生成测试用例,比如 @pytest.mark.parametrize( ['a', 'b', 'exp ...

  7. Day04 - Python 迭代器、装饰器、软件开发规范

    1. 列表生成式 实现对列表中每个数值都加一 第一种,使用for循环,取列表中的值,值加一后,添加到一空列表中,并将新列表赋值给原列表 >>> a = [0, 1, 2, 3, 4, ...

  8. python中的闭包和装饰器

    重新学习完了函数,是时候将其中的一些重点重新捋一捋了,本次总结的东西只有闭包和装饰器 1.闭包 闭包是python函数中的一个比较重要功能,一般闭包都是用在装饰器上,一般学完闭包就会去学习装饰器,这俩 ...

  9. JAVASCRIPT中装饰器是什么(装修)

    装饰器是什么? 解码器是将另一段代码包装在一个代码中的简单方法. 这个概念类似于你以前听说过的功能成分和高阶成分. 这在许多情况下都被使用过,也就是说,成都装修公司简单地将一个函数包装到另一个函数中: ...

随机推荐

  1. SqlLite 简明教程

    SQL DML 和 DDL        可以把 SQL 分为两个部分:数据操作语言 (DML) 和 数据定义语言 (DDL).         注:"--"双减号为行注释     ...

  2. Linux下使用w命令和uptime命令查看系统负载

    在Linux系统中查询系统CPU和内存的负载(使用率)时,我们通常习惯于使用top.atop或者ps,这篇文章将要给大家介绍如何使用w命令和uptime命令来查看系统的负载情况,对于uptime命令, ...

  3. IOS 播放动态Gif图片

    图片分为静态和动态两种,图片的格式有很多种,在开发中比较常见的是.png和.jpg的静态图片,但有的时候在App中需要播放动态图片,比如.gif格式的小表情头像,在IOS中并没有提供直接显示动态图片的 ...

  4. jersey + tomcat 实现restful风格

    本文参考 http://www.cnblogs.com/bluesfeng/archive/2010/10/28/1863816.html 环境: idea 15.0.2 jersey 1.3 tom ...

  5. bluetooth-蓝牙事件监听

    今天在做项目的时候,需要监听一个蓝牙耳机的连接状态.就写了一个小的测试方法.记录如下 看代码 这要处理的是蓝牙监听事件 package com.example.alert; import androi ...

  6. GOOGLE搜索從入門到精通V4.0

    1,前言2,摘要3,如何使用本文4,Google簡介5,搜索入門6,初階搜索 6.1,搜索結果要求包含兩個及兩個以上關鍵字 6.2,搜索結果要求不包含某些特定資訊 6.3,搜索結果至少包含多個關鍵字中 ...

  7. 11.3 morning

    noip模拟题day1 总览(Overview)   题目名称 取模 等比数列 回文串 程序名 mod sequence palindromes 输入文件名 mod.in sequence.in pa ...

  8. Linux命令行编辑快捷键

    Linux命令行编辑快捷键: history 显示命令历史列表 ↑(Ctrl+p) 显示上一条命令 ↓(Ctrl+n) 显示下一条命令 !num 执行命令历史列表的第num条命令 !! 执行上一条命令 ...

  9. web前端开发随手笔记 - 持续更新

    本文仅为个人常用代码整理,供自己日常查阅 html 浏览器内核 <!--[if IE]><![endif]--> <!--[if IE 6]><![endif ...

  10. Linux gvim windows 版本配置

    http://www.cnblogs.com/xiekeli/archive/2012/08/13/2637176.html 资源在我的网盘里面