Итак, вы хотите использовать инфраструктуру типов контента в своей работе?
Начните с того, что задайте себе вопрос: «Должны ли какие-либо из этих моделей быть связаны таким же образом с другими моделями, и / или я буду использовать эти отношения непредвиденными способами позже в будущем?» Причина, по которой мы задаем этот вопрос, заключается в том, что именно это делает структура типов контента лучше всего: она создает общие отношения между моделями. Бла-бла, давайте углубимся в некоторый код и посмотрим, что я имею в виду.
# ourapp.models
from django.conf import settings
from django.db import models
# Assign the User model in case it has been "swapped"
User = settings.AUTH_USER_MODEL
# Create your models here
class Post(models.Model):
author = models.ForeignKey(User)
title = models.CharField(max_length=75)
slug = models.SlugField(unique=True)
body = models.TextField(blank=True)
class Picture(models.Model):
author = models.ForeignKey(User)
image = models.ImageField()
caption = models.TextField(blank=True)
class Comment(models.Model):
author = models.ForeignKey(User)
body = models.TextField(blank=True)
post = models.ForeignKey(Post)
picture = models.ForeignKey(Picture)
Итак, у нас есть способ теоретически создать эти отношения. Однако, как программист Python, ваш превосходный интеллект говорит вам, что это отстой, и вы можете добиться большего. Дай пять!
Войдите в структуру типов контента!
Что ж, теперь мы рассмотрим наши модели и доработаем их, чтобы сделать их более «многоразовыми» и интуитивно понятными. Давайте начнем с того, что избавимся от двух внешних ключей в нашей Comment
модели и заменим их на GenericForeignKey
.
# ourapp.models
from django.contrib.contenttypes.fields import GenericForeignKey
from django.contrib.contenttypes.models import ContentType
...
class Comment(models.Model):
author = models.ForeignKey(User)
body = models.TextField(blank=True)
content_type = models.ForeignKey(ContentType)
object_id = models.PositiveIntegerField()
content_object = GenericForeignKey()
Так что же случилось? Ну, мы вошли и добавили необходимый код, чтобы учесть общее отношение к другим моделям. Обратите внимание , как есть больше , чем просто GenericForeignKey
, но и ForeignKey
для ContentType
и PositiveIntegerField
для object_id
. Эти поля предназначены для сообщения Django, к какому типу объекта это относится и каков id этого объекта. В действительности это имеет смысл, потому что Django понадобится оба для поиска этих связанных объектов.
Ну, это не очень похоже на Python ... это ужасно!
Вы, вероятно, ищете герметичный, безупречный, интуитивно понятный код, который бы гордил Гвидо ван Россума . Я понимаю тебя. Давайте посмотрим на GenericRelation
поле, чтобы мы могли хорошенько поклониться этому.
# ourapp.models
from django.contrib.contenttypes.fields import GenericRelation
...
class Post(models.Model):
author = models.ForeignKey(User)
title = models.CharField(max_length=75)
slug = models.SlugField(unique=True)
body = models.TextField(blank=True)
comments = GenericRelation('Comment')
class Picture(models.Model):
author = models.ForeignKey(User)
image = models.ImageField()
caption = models.TextField(blank=True)
comments = GenericRelation('Comment')
Бам! Точно так же вы можете работать с комментариями для этих двух моделей. На самом деле, давайте продолжим и сделаем это в нашей оболочке (введите python manage.py shell
из вашей директории проекта Django).
>>> from django.contrib.auth import get_user_model
>>> from ourapp.models import Picture, Post
# We use get_user_model() since we are referencing directly
User = get_user_model()
# Grab our own User object
>>> me = User.objects.get(username='myusername')
# Grab the first of our own pictures so we can comment on it
>>> pic = Picture.objects.get(author=me)
# Let's start making a comment for our own picture
>>> pic.comments.create(author=me, body="Man, I'm cool!")
# Let's go ahead and retrieve the comments for this picture now
>>> pic.comments.all()
[<Comment: "Man, I'm cool!">]
# Same for Post comments
>>> post = Post.objects.get(author=me)
>>> post.comments.create(author=me, body="So easy to comment now!")
>>> post.comments.all()
[<Comment: "So easy to comment now!"]
Это так просто.
Каковы другие практические последствия этих «общих» отношений?
Общие внешние ключи обеспечивают менее навязчивые отношения между различными приложениями. Например, допустим, мы вытащили модель Comment из собственного приложения с именем chatterly
. Теперь мы хотим создать другое приложение с именем, в noise_nimbus
котором люди хранят свою музыку, чтобы поделиться с другими.
Что если мы хотим добавить комментарии к этим песням? Ну, мы можем просто нарисовать общее отношение:
# noise_nimbus.models
from django.conf import settings
from django.contrib.contenttypes.fields import GenericRelation
from django.db import models
from chatterly.models import Comment
# For a third time, we take the time to ensure custom Auth isn't overlooked
User = settings.AUTH_USER_MODEL
# Create your models here
class Song(models.Model):
'''
A song which can be commented on.
'''
file = models.FileField()
author = models.ForeignKey(User)
title = models.CharField(max_length=75)
slug = models.SlugField(unique=True)
description = models.TextField(blank=True)
comments = GenericRelation(Comment)
Я надеюсь , что вы , ребята нашли эту информацию полезной , как я любил бы иметь попадались то , что показал мне более реалистичное применение GenericForeignKey
и GenericRelation
полей.
Это слишком хорошо, чтобы быть правдой?
Как и во всем в жизни, есть плюсы и минусы. Каждый раз, когда вы добавляете больше кода и больше абстракции, базовые процессы становятся тяжелее и немного медленнее. Добавление общих отношений может немного снизить производительность, несмотря на то, что она попытается кэшировать свои результаты. В общем, все сводится к тому, перевешивает ли чистота и простота небольшие эксплуатационные расходы. Для меня ответ - миллион раз да.
В структуре типов контента есть нечто большее, чем я показал здесь. Существует целый уровень детализации и более подробное использование, но для среднего человека, по моему мнению, вы будете использовать его 9 из 10 раз.
Общие реляционеры (?) Будьте осторожны!
Довольно большое предостережение заключается в том, что при использовании a GenericRelation
, если удаляется модель, имеющая GenericRelation
apply ( Picture
), все связанные Comment
объекты ( ) также будут удалены. Или, по крайней мере, на момент написания этой статьи.