Одним из способов значительно ускорить скорость работы сайта на Django - это кэширование как отдельных частей шаблонов сайта, так и кэширование шаблонов после их компилирования сайтом. Поэтому изучим оба этих способа улучшения скорости работы сайта, помимо уже известного нам способа правильной оптимизации запросов к базе данных Django . Проверять результативность улучшений можно по прежнему с помощью батарейки django-silk, что и описано в статье по улучшению запросов к базе данных.
А теперь расмотрим варианты использования кэширования.
Кэширование частей шаблонов
В тех случаях, когда у вас есть генерируемые части шаблонов, как например нижний колонтитул сайта (Footer) или боковые панели (SideBar), то моэно использовать кэширование этих частей сайта. Например, я закэшировал не только их, но даже и Navigation Drawer сайта и верхнюю панель навигации. Сделано это было по той причине, что я использую для генерирования ссылок навигации встраиваимые теги url а также теги перевода trans , которых набирается большое количество, и в совокупности они добавляют прилчную нагрузку на сайт. Не говоря уже о том, что у меня используются динамически настраиваемые виджеты (об этом вы можете прочитать в статье о полиморфной системе динамических виджетов . И именно эти виджеты при каждой загрузке могут удвоить или даже утроить время генерирования страницы сайта, а если учесть, что виды виджетов могут быть самыми разными, то и написать достаточно эффективный запрос к базе данных уже не получится. Поэтому проще всего закшировать эти части шаблона сайта.
Это будет выглядеть следующим образом
- {% load i18n cache %}
- {% get_current_language as LANGUAGE_CODE %}
- {% cache 6000 sidebar LANGUAGE_CODE %}
- {% load sidebar sidebar_sticky from evileg_widgets %}
- {% sidebar %}
- {% sidebar_sticky %}
- {% endcache %}
В данном коде используется встраиваемый тег cache, в который передаётся ключ кеша sidebar , а также в качестве дополнительного параметра код языка LANGUAGE_CODE . Код языка неоходим, чтобы поддерживать мультиязычность сайта.
В данном случае с динамическими виджетами возникает важный нюанс, а именно инвалидация кэша в том случае, если SideBar с динамическими виджетам был изменён через административную панель сайта.
Так вот, это реализовал следующим образом:
- # -*- coding: utf-8 -*-
- from django.conf import settings
- from django.core.cache import cache
- from django.db import models
- from django.core.cache.utils import make_template_fragment_key
- from django.db.models.signals import post_save, post_delete
- from solo.models import SingletonModel
- # Some another imports
- class SideBar(SingletonModel):
- # Some code of model
- class Widget(models.Model):
- # Some code of model
- def invalidate_cache(**kwargs):
- for code, description in settings.LANGUAGES:
- cache.delete(make_template_fragment_key('sidebar', [code]))
- post_save.connect(invalidate_cache, sender=SideBar)
- post_save.connect(invalidate_cache, sender=Widget)
- post_delete.connect(invalidate_cache, sender=Widget)
Как видите из кода, здесь используется система сигналов, что к слову очень похоже на систему сигналов и слотов в Qt , что мне как Qt разработчику очень даже понравилось.
Так вот, данная система сигналов и слотов инвалидует кэш в том случае, если был изменён как сам объект SideBar, так и в какой-то из виджетов. Причём инвалидация происходит сразу для всех языков. Таким образом тяжелые запросы к базе данных возникают не чаще, чем раз в 6000 секунд. Что вполне приемлемо.
Кэшированый загрузчик шаблонов
А следующим способом повышения производительности сайта является использование кэшированного загрузчика шаблонов. Дело в том, что обычно Django ищет шаблоны при каждом обращении к странице сайта, если же настроить кэширование у загрузчика шаблонов, то производительность сайта может вырасти вдвое.
Например, у меня время генерирования некоторых страниц сайта после включения данной опции уменьшилось с 220-240 мс до 120-130. Конечно это с учётом и многих других ухищрений для повыщения производительности. Но тем не менее результат очень хороший.
А настройка данного функционала производится в файле settigns.py и выглядеть это должно следующим образом.
- TEMPLATES = [
- {
- 'BACKEND': 'django.template.backends.django.DjangoTemplates',
- 'DIRS': [ os.path.join(BASE_DIR, 'templates') ],
- 'OPTIONS': {
- # some another options
- 'loaders': [
- ('django.template.loaders.cached.Loader', [
- 'django.template.loaders.filesystem.Loader',
- 'django.template.loaders.app_directories.Loader',
- ]),
- ],
- },
- },
- ]
Заключение
Очень рекомендую закэшировать части шаблона, даже если там всего лишь вызовы тегов url или trans , в глобальной перспективе это тоде может повлиять на качество вашего сайта для поисковых систем. Ведь не зря в интернете так много статей, в которых говорится, что поисковики повышают в выдаче очень быстрые сайты.