django是一个python web开发的框架。作为一个框架MVC的架构已经实现起来了。但是编码的时候你经常要进行进一步的抽象。

AOP是一种称为面向切面的开发思想,意思是将部分功能代码在运行时动态的加载到指定位置。最常见的应用是Spring中的依赖注入@Autowired。

而装饰器也可以被看成是一种AOP的实现,但是又有些许的不同,让我们来体会一下。

在我们的实例中我们将django中的views.py(其实是controller层)拆出了implement.py(实现)和decorator.py(装饰器)

1、先看我们的views.py。

这里我们只给出一个接口做为演示。

# -*- coding:utf-8 -*-
import json
import logging
import implement
import decorator
from django.views.decorators.csrf import csrf_exempt logger = logging.getLogger(__name__) @csrf_exempt
@decorator.impl_wrapper_check_time
@decorator.process_time
@decorator.cache
def alarm_month_daytotal(request):
if request.method == 'GET':
time_str = request.GET.get('time')
return implement.alarm_month_daytotal(time_str)

alarm_total接口在views.py中的函数alarm_total中并没有返回JSONResponse对象,而是直接返回了调用implement.alarm_total()的结果,看样子是implement.alarm_total()返回了JSONResponse

,我们看看是不是这样。

2、implement实现层

# -*- coding:utf-8 -*-
import json
import logging
import calendar from rediss import RedisCache
from config import redis_config, alarm_status_mapping, alarm_type_mapping, alarm_level_mapping, combine_module, alarm_module_mapping,alarm_stage_mapping, \
alarm_type_config_mapping, month_total_closed_mapping, month_total_tosolve_mapping
from datetime import datetime, timedelta
from dao import Dao as dash_dao
from exceptions import DataEmptyException
import time logger = logging.getLogger(__name__) # 按月获取分天报警情况
def alarm_month_daytotal(time_str):
time_object = datetime.strptime(time_str, '%Y-%m')
month, length = calendar.monthrange(time_object.year, time_object.month)
ret_list = [0 for i in range(length)]
items = dash_dao.get_alarms_by_month(time_str)
if not items:
raise DataEmptyException('[table] %s' % 'alarm_list_table')
for item in items:
if not item.alarm_time:
continue
ret_list[at_day(item.alarm_time) - 1] += 1
r = RedisCache(redis_config)
key_list = r.keys("dmonitor:issue:%s*" % time_str)
if not key_list:
return ret_list
for key in key_list:
content = r.get(key)
time_object = datetime.strptime(key.split(':')[2], '%Y-%m-%d')
ret_list[time_object.day - 1] = {
'y': ret_list[time_object.day - 1],
'name': content,
'marker': {
'symbol': 'url(/data_monitor/static/images/sun.png)'
}
}
return ret_list

并没有啊,implement.alarm_total()只返回了一个list对象。这是为什么呢?

原因就在那几个装饰器的封装上。

3、装饰器decorator

impl_wrapper_check_time(func):执行装饰的装饰的func方法,并且对返回进行JSONResponse封装

process_time(func):统计执行时间并打印日志

cache(func):对接口的请求加入缓存

执行顺序,装饰器装饰的顺序,从下往上我们例子里是cache->process_time->impl_wrapper_check_time那么:

1、先执行impl_wrapper_check_time的开始部分

2、然后是process_time时间的start_time记录

3、cache缓存的准备

4、被装饰的函数func

5、cache缓存的返回

6、process_time的end_time记录,并打印时间日志

7、impl_wrapper_check_time返回JSONResponse

执行顺序说明:

1、异常也是按照这个顺序一级一级的向上抛出。

2、最终由impl_wrapper_check_time处理异常,返回errno:0或者-1。

3、先执行完缓存的返回,再执行时间的统计,这样可以明显观察到缓存对处理时间性能上的提升。

import logging
import time
import json
import traceback
from rediss import RedisCache
from config import redis_config, cache_timeout, cache_switch
from exceptions import IllegalParamException
from django.http import JsonResponse
from django.http import HttpResponse logger = logging.getLogger(__name__) redis = RedisCache(redis_config)def impl_wrapper_check_time(func):
def wrapper(*args, **kwargs):
try:
if args[0].method == 'GET':
if not args[0].GET.get('time'):
raise IllegalParamException('time')
data = func(*args, **kwargs)
return JsonResponse({'errno': 0, 'msg': 'success', 'data': data})
except Exception, ex:
logger.error(traceback.format_exc())
return JsonResponse({'errno': -1, 'msg': str(ex)}) return wrapper def process_time(func):
def wrapper(*args, **kwargs):
path = args[0].get_full_path()
start_time = time.time()
data = func(*args, **kwargs)
end_time = time.time()
logger.info('path: %s, process_time: %s ms' % (path, str((end_time - start_time) * 1000)))
return data
return wrapper def cache(func):
def wrapper(*args, **kwargs):
if not cache_switch:
data = func(*args, **kwargs)
return data
path = args[0].get_full_path()
dashboard_cache_key = 'dashboard:cache:%s' % path
if redis.get(dashboard_cache_key):
logger.info('[Hit Cache] path: %s' % path)
return json.loads(redis.get(dashboard_cache_key))
data = func(*args, **kwargs)
redis.set(dashboard_cache_key, json.dumps(data))
redis.expire(dashboard_cache_key, cache_timeout)
logger.info('[Query] path: %s' % path)
return data
return wrapper

python使用装饰器@函数式化django开发的更多相关文章

  1. Python 之装饰器

    Python 的装饰器可谓是提高开发效率的一大利器.然而初学装饰器的时候感觉很难理解,因为除了 Python 之外没听说哪个语言有这种东西. 而且网上看的很多解释看似容易理解,但只能很快理解了装饰器能 ...

  2. 浅谈Django的中间件与Python的装饰器

    浅谈Django的中间件 与Python的装饰器 一.原理 1.装饰器是Python的一种语法应用,利用闭包的原理去更改一个函数的功能,即让一个函数执行之前先到另外一个函数中执行其他需求语句,在执行该 ...

  3. Python各式装饰器

    Python装饰器,分两部分,一是装饰器本身的定义,一是被装饰器对象的定义. 一.函数式装饰器:装饰器本身是一个函数. 1.装饰函数:被装饰对象是一个函数 [1]装饰器无参数: a.被装饰对象无参数: ...

  4. 详解Python的装饰器

    Python中的装饰器是你进入Python大门的一道坎,不管你跨不跨过去它都在那里. 为什么需要装饰器 我们假设你的程序实现了say_hello()和say_goodbye()两个函数. def sa ...

  5. 进阶Python:装饰器 全面详解

    进阶Python:装饰器 前言 前段时间我发了一篇讲解Python调试工具PySnooper的文章,在那篇文章开始一部分我简单的介绍了一下装饰器,文章发出之后有几位同学说"终于了解装饰器的用 ...

  6. 我终于弄懂了Python的装饰器(一)

    此系列文档: 1. 我终于弄懂了Python的装饰器(一) 2. 我终于弄懂了Python的装饰器(二) 3. 我终于弄懂了Python的装饰器(三) 4. 我终于弄懂了Python的装饰器(四) 一 ...

  7. 我终于弄懂了Python的装饰器(四)

    此系列文档: 1. 我终于弄懂了Python的装饰器(一) 2. 我终于弄懂了Python的装饰器(二) 3. 我终于弄懂了Python的装饰器(三) 4. 我终于弄懂了Python的装饰器(四) 四 ...

  8. Python札记 -- 装饰器补充

    本随笔是对Python札记 -- 装饰器的一些补充. 使用装饰器的时候,被装饰函数的一些属性会丢失,比如如下代码: #!/usr/bin/env python def deco(func): def ...

  9. python基础——装饰器

    python基础——装饰器 由于函数也是一个对象,而且函数对象可以被赋值给变量,所以,通过变量也能调用该函数. >>> def now(): ... print('2015-3-25 ...

随机推荐

  1. centos 7 && dotnet core 2.0 && nginx && supervisor

    前提 系统:centos 7 目录:/home/wwwroot/www.wuball.com dotnet core 2.0 官方指引 sudo rpm --import https://packag ...

  2. centos6.6配置rsync+sersync实现实时同步分布式多客户端分发同步

    1.sersync项目: sersync项目利用inotify与rsync技术实现对服务器数据实时同步到解决方案,其中inotify用于监控sersync所在服务器上文件系统的事件变化,rsync是目 ...

  3. 团队作业4--第一次项目冲刺(Alpha版本) 4

    一.Daily Scrum Meeting照片 二.燃尽图 三.项目进展 完成对查重结果的写出与保存,将查重结果写出并导出保存为Excel形式 四.困难与问题 对查重结果的保存,当有多份文档进行比较的 ...

  4. 团队作业8----第二次项目冲刺(beta阶段)5.23

    Day5-05.23 1.每日会议 会议内容: 1.组长林乔桦对昨日的工作进行了总结并且安排今日的任务. 2.潘益靖副组长说明昨日任务的完成情况. 3.组员对昨天的各项工作进行了汇报以及对今天的工作进 ...

  5. 201521123024《Java程序设计》第3周学习总结

    1. 本周学习总结 2. 书面作业 1.代码阅读 public class Test1 { private int i = 1;//这行不能修改 private static int j = 2; p ...

  6. Java课程设计博客(个人)

    Java课程设计博客(个人) 1. 团队课程设计博客链接 http://www.cnblogs.com/wkfg/p/7063081.html 2. 个人负责模块或任务说明 负责模块/任务:编写doG ...

  7. 201521123023《Java程序设计》第10周学习总结

    1. 本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结异常与多线程相关内容. 2. 书面作业 本次PTA作业题集异常.多线程 1.finally 题目4-2 1.1 截图你的提交结果(出 ...

  8. 【Intellij Idea】设置JDK

    1,File-->Project Structure 说明:可以设置整个工程的JDK,也设置每个modules的JDK,但是一般整个工程的JDK. 2,或者通过快捷键:ctrl+alt+shif ...

  9. linux(5)--补充(管道| / 重定向> / xargs)/find 与xargs结合使用/vi,grep,sed,awk(支持正则表达式的工具程序)

    本节中正则表达式的工具程序 grep,sed和awk是重点,也是难点!!! 先补充一下一. 管道| / 重定向> / xargs 如:1. 管道和重定向的区别:具体可以见 http://www. ...

  10. JavaScript配合button.onclick()使用总结

    Html加载顺序是从上往下加载,如果script中含有引用js脚本,则应该将此script放在head标签里面,这样可是保证此页面都可以引用js脚本内容.如果想在script中设置button.onc ...