【python】-- Django 中间件、缓存、信号
Django? 中間件、緩存、信號
一、 Django? 中間件
django 中的中間件(middleware),在django中,中間件其實就是一個類,在請求到來和結束后,django會根據自己的規則在合適的時機執行中間件中相應的方法。
在django項目的settings模塊中,有一個 MIDDLEWARE_CLASSES?變量,其中每一個元素就是一個中間件:
MIDDLEWARE = ['django.middleware.security.SecurityMiddleware','django.contrib.sessions.middleware.SessionMiddleware','django.middleware.common.CommonMiddleware','django.middleware.csrf.CsrfViewMiddleware','django.contrib.auth.middleware.AuthenticationMiddleware','django.contrib.messages.middleware.MessageMiddleware','django.middleware.clickjacking.XFrameOptionsMiddleware', ]1、中間件中可以定義四個方法:
process_request(self,request) #每個進入view視圖中請求都要經過這個函數 process_response(self, request, response) #每個從view視圖中返回的response都會觸發這個函數 process_view(self, request, callback, callback_args, callback_kwargs) #經過路由映射找到view中對應的方法后,就會觸發這個函數,找不到就不會觸發 process_template_response(self,request,response) # 如果Views中的函數返回的對象中,具有render方法,這個函數才會執行 process_exception(self, request, exception) #如果對應的視圖中的函數出現異常,這個函數才會執行2、自定義中間件
與mange.py在同一層級下新建目錄Middle,在Middle新建m1.py(文件名稱隨意):
1、在m1.py中定義三個中間件
class Foo:# render 方法用來觸發process_template_response函數def render(self):return HttpResponse("template_response")def test(request):print("窮")return Foo() views.py from django.utils.deprecation import MiddlewareMixin # 自定義中間件都需要繼承這個類# 定義中間件Row1 class Row1(MiddlewareMixin):def process_request(self,request): # 所有經過中間件Row1的request都會打印001print('request——001')# request這個參數是請求的request, view_func指定的視圖中對應的函數, view_func_args路由系統中的位置參數, view_func_kwargs路由系統中的關鍵字參數def process_view(self, request, view_func, view_func_args, view_func_kwargs):print('view1')def process_response(self, request, response): # 所有經過中間件Row1的response 都會打印001print('response——001')return responsefrom django.shortcuts import HttpResponse# 定義中間件Row2 class Row2(MiddlewareMixin):def process_request(self,request):print('request——002')# 如果不想讓request進入view視圖函數中,可以在自定義中間件中直接return返回。# 終止之后就不會經過row3的response,會直接從row2 response返回, 這個特性是1.7之后才有的。1.7以前會經過row3的response之后再進行返回。# return HttpResponse('終止')def process_view(self, request, view_func, view_func_args, view_func_kwargs):print('view2')def process_response(self, request, response):print('response——002')return response# 定義中間件Row3 class Row3(MiddlewareMixin):def process_request(self,request):print('request——003')def process_view(self, request, view_func, view_func_args, view_func_kwargs):print('view2')def process_response(self, request, response):print('response——003')return response# 這個函數的觸發條件就是如果對應的視圖中的函數出現異常,這個函數才會執行def process_exception(self, request, exception):if isinstance(exception,ValueError):return HttpResponse('出現異常》。。')# 如果Views中的函數返回的對象中,具有render方法,這個函數才會執行,像redirect,HttpResponse 都不會觸犯這個函數def process_template_response(self,request,response):print('-----------------------')return response2、注冊中間件:
在settings文件中的MIDDLEWARE,注冊剛剛定義的三個中間件:
MIDDLEWARE = ['django.middleware.security.SecurityMiddleware','django.contrib.sessions.middleware.SessionMiddleware','django.middleware.common.CommonMiddleware','django.middleware.csrf.CsrfViewMiddleware','django.contrib.auth.middleware.AuthenticationMiddleware','django.contrib.messages.middleware.MessageMiddleware','django.middleware.clickjacking.XFrameOptionsMiddleware','Middle.m1.Row1','Middle.m1.Row2','Middle.m1.Row3', ]3、中間件request、response狀態流程示意圖:
?
?
二、Django? 緩存
由于Django是動態網站,所有每次請求均會去數據進行相應的操作,當程序訪問量大時,耗時必然會更加明顯,最簡單解決方式是使用:緩存,緩存將一個某個views的返回值保存至內存或者memcache中,5分鐘內再有人來訪問時,則不再去執行view中的操作,而是直接從內存或者Redis中之前緩存的內容拿到,并返回,Django是目前僅有自帶緩存的框架。
Django中提供了7種緩存方式:
- 開發調試
- 內存
- 文件
- 數據庫
- Memcache緩存(python-memcached模塊)
- Memcache緩存(pylibmc模塊)
- Redis緩存
一、緩存配置方式
1、配置
a、開發調試
# 此為開始調試用,實際內部不做任何操作# 配置:CACHES = {'default': {'BACKEND': 'django.core.cache.backends.dummy.DummyCache', # 引擎'TIMEOUT': 300, # 緩存超時時間(默認300,None表示永不過期,0表示立即過期)'OPTIONS':{'MAX_ENTRIES': 300, # 最大緩存個數(默認300)'CULL_FREQUENCY': 3, # 緩存到達最大個數之后,剔除緩存個數的比例,即:1/CULL_FREQUENCY(默認3) },'KEY_PREFIX': '', # 緩存key的前綴(默認空)'VERSION': 1, # 緩存key的版本(默認1)'KEY_FUNCTION' 函數名 # 生成key的函數(默認函數會生成為:【前綴:版本:key】) }}# 自定義keydef default_key_func(key, key_prefix, version):"""Default function to generate keys.Constructs the key used by all other methods. By default it prependsthe `key_prefix'. KEY_FUNCTION can be used to specify an alternatefunction with custom key making behavior."""return '%s:%s:%s' % (key_prefix, version, key)def get_key_func(key_func):"""Function to decide which key function to use.Defaults to ``default_key_func``."""if key_func is not None:if callable(key_func):return key_funcelse:return import_string(key_func)return default_key_func View Codeb、內存
# 此緩存將內容保存至內存的變量中# 配置:CACHES = {'default': {'BACKEND': 'django.core.cache.backends.locmem.LocMemCache','LOCATION': 'unique-snowflake', #變量值要唯一 }}# 注:其他配置同開發調試版本 View Codec、文件
# 此緩存將內容保存至文件# 配置: CACHES = {'default': {'BACKEND': 'django.core.cache.backends.filebased.FileBasedCache','LOCATION': '/var/tmp/django_cache',}}# 注:其他配置同開發調試版本 View Coded、數據庫
# 此緩存將內容保存至數據庫# 配置:CACHES = {'default': {'BACKEND': 'django.core.cache.backends.db.DatabaseCache','LOCATION': 'my_cache_table', # 數據庫表 }}# 注:執行創建表命令 python manage.py createcachetable View Codee、Memcache緩存(python-memcached模塊)
# 此緩存使用python-memcached模塊連接memcache CACHES = {'default': {'BACKEND': 'django.core.cache.backends.memcached.MemcachedCache','LOCATION': '127.0.0.1:11211',}}CACHES = {'default': {'BACKEND': 'django.core.cache.backends.memcached.MemcachedCache','LOCATION': 'unix:/tmp/memcached.sock',}} CACHES = {'default': {'BACKEND': 'django.core.cache.backends.memcached.MemcachedCache','LOCATION': ['172.19.26.240:11211','172.19.26.242:11211',]}} View Codef、Memcache緩存(pylibmc模塊)
# 此緩存使用pylibmc模塊連接memcache CACHES = {'default': {'BACKEND': 'django.core.cache.backends.memcached.PyLibMCCache','LOCATION': '127.0.0.1:11211',}}CACHES = {'default': {'BACKEND': 'django.core.cache.backends.memcached.PyLibMCCache','LOCATION': '/tmp/memcached.sock',}} CACHES = {'default': {'BACKEND': 'django.core.cache.backends.memcached.PyLibMCCache','LOCATION': ['172.19.26.240:11211','172.19.26.242:11211',]}} View Codeg. Redis緩存(依賴:pip3 install django-redis)
CACHES = {"default": {"BACKEND": "django_redis.cache.RedisCache","LOCATION": "redis://127.0.0.1:6379","OPTIONS": {"CLIENT_CLASS": "django_redis.client.DefaultClient","CONNECTION_POOL_KWARGS": {"max_connections": 100}# "PASSWORD": "密碼", }} } View Code from django_redis import get_redis_connection conn = get_redis_connection("default") 視圖種連接并操作2、緩存的三種方式:
使用中間件,經過一系列的認證等操作,如果內容在緩存中存在,則使用FetchFromCacheMiddleware獲取內容并返回給用戶,當返回給用戶之前,判斷緩存中是否已經存在,如果不存在則UpdateCacheMiddleware會將緩存保存至緩存,從而實現全站緩存MIDDLEWARE = ['django.middleware.cache.UpdateCacheMiddleware', #這個要設置在最上層,經過所有中間件的response認證之后再把緩存內容返回給用戶# 其他中間件...'django.middleware.cache.FetchFromCacheMiddleware', #這個中間件要設置在所有中間件最下層,比如如果設置CSRF中間件前,request請求不用通過csrf認證就能拿到返回 ]CACHE_MIDDLEWARE_ALIAS = ""CACHE_MIDDLEWARE_SECONDS = ""CACHE_MIDDLEWARE_KEY_PREFIX = "" 全局設置緩存 方式一:from django.views.decorators.cache import cache_page@cache_page(60 * 15) #通過裝飾器,單位是秒def my_view(request):...方式二:from django.views.decorators.cache import cache_pageurlpatterns = [url(r'^foo/([0-9]{1,2})/$', cache_page(60 * 15)(my_view)),] 單獨視圖緩存 a. 引入TemplateTag{% load cache %}b. 使用緩存{% cache 5000 緩存key %} # 時間以秒為單位,緩存key自定義 緩存內容{% endcache %} 單獨模板緩存三種緩存方式的優先級:如果三種緩存方式同時存在的話,全局設置緩存>單獨視圖緩存>單獨模板緩存, 優先級跟請求生命周期有關
3、文件緩存示例:
a、settings.py中配置
CACHES = {'default': {'BACKEND': 'django.core.cache.backends.filebased.FileBasedCache','LOCATION': os.path.join(BASE_DIR,'cache') #cache文件的保存路徑 } }b、templates下的html文本
{% load cache %} #導入緩存模塊 <!DOCTYPE html> <html lang="en"> <head><meta charset="UTF-8"><title></title> </head> <body><h1>{{ ctime }}</h1><h1>{{ ctime }}</h1>{% cache 10 k1 %} #緩存10秒<h1>{{ ctime }}</h1>{% endcache %} </body> </html> cache.htmlc、views.py中的函數
from django.views.decorators.cache import cache_page@cache_page(10) #緩存10秒 def cache(request):import timectime = time.time()return render(request, 'cache.html', {'ctime': ctime})?更多Django緩存內容:點擊
?
三、Django 信號
Django中提供了“信號調度”,用于在框架執行操作時解耦。通俗來講,就是一些動作發生的時候,信號允許特定的發送者去提醒一些接受者。
1、Django內置信號
Model signalspre_init # django的modal執行其構造方法前,自動觸發post_init # django的modal執行其構造方法后,自動觸發pre_save # django的modal對象保存前,自動觸發post_save # django的modal對象保存后,自動觸發pre_delete # django的modal對象刪除前,自動觸發post_delete # django的modal對象刪除后,自動觸發m2m_changed # django的modal中使用m2m字段操作第三張表(add,remove,clear)前后,自動觸發class_prepared # 程序啟動時,檢測已注冊的app中modal類,對于每一個類,自動觸發 Management signalspre_migrate # 執行migrate命令前,自動觸發post_migrate # 執行migrate命令后,自動觸發 Request/response signalsrequest_started # 請求到來前,自動觸發request_finished # 請求結束后,自動觸發got_request_exception # 請求異常后,自動觸發 Test signalssetting_changed # 使用test測試修改配置文件時,自動觸發template_rendered # 使用test測試渲染模板時,自動觸發 Database Wrappersconnection_created # 創建數據庫連接時,自動觸發內置信號的觸發示例:
a.創建信號觸發方式
在manage.py同級目錄下創建sg.py文件(路徑可自定義)
# Django 所有內置函數的導入方式 from django.core.signals import request_finished from django.core.signals import request_started from django.core.signals import got_request_exceptionfrom django.db.models.signals import class_prepared from django.db.models.signals import pre_init, post_init from django.db.models.signals import pre_save, post_save from django.db.models.signals import pre_delete, post_delete from django.db.models.signals import m2m_changed from django.db.models.signals import pre_migrate, post_migratefrom django.test.signals import setting_changed from django.test.signals import template_renderedfrom django.db.backends.signals import connection_created# 自定義一個函數 def f1(sender, **kwargs): print("hello")# print(sender,kwargs)# 內置信號關聯觸發函數 pre_save.connect(f1)b.讓view中的函數觸發信號
需要在與工程同名的目錄下的__init__.py文件中導入剛剛創建的sg.py文件
import sgc 、view.py中進行save操作:
def signal(reuqest):from app01 import modelsobj = models.UserInf(user='root')print('start')obj.save()obj = models.UserInf(user='root')obj.save()obj = models.UserInf(user='root')obj.save()在每個save操作后,信號都會觸發一次觸發函數,打印一次hello # 輸出: start hello hello hello2、自定義信號
a.sg.py中自定義信號
import django.dispatch # 導入自定義信號類# pizza_done 信號名可以自定義 providing_args=["toppings", "size"] 這個設置的是pizza_done信號的觸發條件 pizza_done = django.dispatch.Signal(providing_args=["toppings", "size"])def f2(sender, **kwargs):print("f2_callback")print(sender, kwargs)# 自定義信號關聯觸發函數 pizza_done.connect(f2)b.__init__.py文件
import sgc 、view.py文件
from sg import pizza_done #導入自定義信號pizza_done.send(sender="kkk",toppings=123, size=456) #sender 名字自定義 return HttpResponse('ok')
轉載于:https://www.cnblogs.com/Keep-Ambition/p/8900453.html
總結
以上是生活随笔為你收集整理的【python】-- Django 中间件、缓存、信号的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 奉献一波鹅厂的面经!纪念最后的校招!|
- 下一篇: Netty 4.x – ByteBuf