- 1. shortcuts.py
- 2. Приклад
У шорткатах Django є функція get_object_or_404 , але при цьому немає функції get_object_or_none , яка може бути корисною в тому випадку, коли ми не хочемо викликати відразу помилку 404, але при цьому нам необхідно виконати якісь дії в тому у випадку, якщо об'єкт не існує в базі даних.
Вдалося знайти обговорення в інтернеті, де розробники відписалися, що подібний функціонал може бути легко впроваджений як until функції в рамках проекту розробника і впроваджувати його в основний код не потрібно.
Подібний функціонал мені знадобився для пошуку тегів на сайті.
Тому я його додав у ядро сайту як шорткат.
shortcuts.py
- # -*- coding: utf-8 -*-
- def get_object_or_none(klass, *args, **kwargs):
- try:
- return klass._default_manager.get(*args, **kwargs)
- except klass.DoesNotExist:
- return None
Приклад
- def get_queryset(self, **kwargs):
- q = self.request.GET.get('q')
- tag = get_object_or_none(Tag, name=q)
- if tag:
- return TaggedItem.objects.all().filter(tag=tag)
- return []
В вашем случае происходит подмена сущностей. Вместо того, чтобы взять один конкретный объект, вы забираете queryset а потом берёте из него первый объект. Нехорошо будет, если queryset в каком-то случае вернёт два объекта или более. Проблема в том, что вы не узнаете об этой ошибке никак, а это нарушит логику работы приложения с последствиями в будущем. Мой вариант решения хотя бы порушит приложение и даст знать что существует два одинаковых по каким-то параметрам объекта, и нужно будет принимать меры для исправления.
Согласен с тем что ваше решение более очевидно при чтении кода. first() же здесь применяется не совсем по назначению. А с последствиями "моего" решения не согласен. Метод вернёт только один объект, если он существует, или None. А целостность базы данных лучше, наверно, проверять на этапе добавления записи, а не при доступе к ней. В остальном поведение идентично вашему методу.
А вы гарантируете, что метод first вернёт нужный объект, если в таблице две похожих записи? Этого никто не гарантирует. Может возникнуть неопределённое поведение приложения, если запись не так, что требовалась. А насчёт целостности базы данных перед добавлением я согласен, хотя по идее уровни блокировки на чтение записи в самой базе данных должны разруливать такую ситуацию, но у меня случалась подобная ситуация. Увы..