django - 如何正确查询 manyTOmany 模型字段?
问题描述
好吧,这就是我作为开发者新手要做的事情。我正在尝试为用户创建一个收件箱。我有一个Conversation
具有多对多字段的模型,然后我有一个InstantMessanger
模型,该模型具有 sender_id、date 和一个 receiver_id 字段,该字段是Conversation
. 我试图做的是迭代Conversation
首先显示与用户进行过 convo 的成员,然后显示一个链接,该链接将您带到该消息的实际内容。然而,我坚持让成员们甚至展示。我在这里尝试了很多东西并环顾四周,但我不断收到错误,例如“对话”模型不可迭代或显示但显示为“dating_app.Profile.none”、对象等。所以我很困在这里,不知道该怎么做。这样做的正确方法是什么?
视图.py
def messages(request,profile_id):
messages = InstantMessage.objects.filter(Q(sender_id=request.user)).\
values('sender_id','receiver_id', 'message', 'date', ).\
order_by('date',)
profile = get_object_or_404(Profile,id=profile_id)
conversations = Conversation.objects.filter(members=request.user)
return render(request, 'dating_app/messages.html', {'messages': messages,'profile': profile,'conversations':conversations,})
消息.html
{% for conversation in conversations %}
<li class="text-right list-group-item">
{% for members in conversation %}{% if members != request.user %}
{{ members.username }}<br/>
{% endif %}{% endfor %}
</li>
{%endfor %}
模型.py
class ProfileManager(BaseUserManager):
def create_user(self, username, email,description,photo, password=None):
if not email:
raise ValueError("You must creat an email")
if not username:
raise ValueError("You must create a username!")
if not description:
raise ValueError("You must write a description")
if not photo:
raise ValueError("You must upload a photo")
user = self.model(
email=self.normalize_email(email),
username = username,
description= description,
photo= photo,
)
user.set_password(password)
user.save(using=self._db)
return user
def create_superuser(self, username, email,description,photo, password):
user = self.create_user(
email=self.normalize_email(email),
password=password,
username=username,
description=description,
photo=photo,
)
user.is_admin=True
user.is_staff=True
user.is_superuser=True
user.save(using=self._db)
return user
class Profile(AbstractBaseUser):
class Meta:
swappable = 'AUTH_USER_MODEL'
email = models.EmailField(verbose_name="email")
username = models.CharField(max_length=30, unique=True)
date_joined = models.DateTimeField(verbose_name='date joined', auto_now_add=True)
last_login = models.DateTimeField(verbose_name='last login', auto_now=True)
is_admin = models.BooleanField(default=False)
is_active = models.BooleanField(default=True)
is_staff = models.BooleanField(default=False)
is_superuser = models.BooleanField(default=False)
#what I added
description = models.TextField()
photo = models.ImageField(upload_to='profile_photo',blank=False, height_field=None, width_field=None, max_length=100)
matches = models.ManyToManyField(settings.AUTH_USER_MODEL, related_name='+', blank=True)
USERNAME_FIELD = 'username'
REQUIRED_FIELDS = ['description','photo','email']
objects = ProfileManager()
def __str__(self):
return self.username
def has_perm(self, perm, obj=None):
return self.is_admin
def has_module_perms(self,app_label):
return True
class UserVote(models.Model):
user = models.ForeignKey(settings.AUTH_USER_MODEL, on_delete=models.CASCADE)
voter = models.ForeignKey(settings.AUTH_USER_MODEL, related_name='given_vote', on_delete=models.CASCADE)
vote = models.BooleanField(default=False)
class Meta:
unique_together = (('user', 'voter'))
class Conversation(models.Model):
members = models.ManyToManyField(settings.AUTH_USER_MODEL)
class InstantMessage(models.Model):
sender = models.ForeignKey(settings.AUTH_USER_MODEL, related_name= 'senderr',on_delete=models.CASCADE )
receiver = models.ForeignKey(Conversation, on_delete=models.CASCADE)
message = models.TextField()
date = models.DateTimeField(auto_now_add=True)
def __unicode__(self):
return self.message
解决方案
您可以通过以下方式显示成员:
{% for conversation in conversations %}
<li class="text-right list-group-item">
{% for member in conversation.members.all %}
{% if member != request.user %}
{{ member.username }}<br/>
{% endif %}
{% endfor %}
<li>
{% endfor %}
然而,这将导致N+1问题,为了显示成员,您将在每个对话中进行额外的查询,这不是很有效。.prefetch_related(..)
您可以使用[Django-doc]提高效率:
def messages(request,profile_id):
messages = InstantMessage.objects.filter(Q(sender_id=request.user)).\
values('sender_id','receiver_id', 'message', 'date', ).\
order_by('date',)
profile = get_object_or_404(Profile,id=profile_id)
conversations = Conversation.objects.filter(
members=request.user
).prefetch_related('members')
return render(request, 'dating_app/messages.html', {'messages': messages,'profile': profile,'conversations':conversations,})
编辑:您可以按最后收到的消息(按降序)对消息进行排序:
from django.db.models import Max
def messages(request,profile_id):
messages = InstantMessage.objects.filter(Q(sender_id=request.user)).\
values('sender_id','receiver_id', 'message', 'date', ).\
order_by('date',)
profile = get_object_or_404(Profile,id=profile_id)
conversations = Conversation.objects.filter(
members=request.user
).annotate(
last_message=Max('instantmessage__date')
).prefetch_related('members').order_by(
'-last_message'
)
return render(request, 'dating_app/messages.html', {'messages': messages,'profile': profile,'conversations':conversations,})
推荐阅读
- reactjs - Redux Store 在 getInitialProps 中填充,但在客户端为空
- python - Python Scrapy - 从 url 抓取的不是 start_urls 中设置的
- flutter - 在没有 SetState 的情况下调用持久性底页(使用 StreamBuilder)
- python - 如何将列表列表转换为字节?
- php - If Else with isset: else 部分不起作用 [PHP]
- javascript - 谷歌浏览器片段:标识符“...”已被声明
- r - 为相关矩阵选择变量
- c# - console.writeline 数组的值在一行上
- jakarta-ee - Jeddict 可以安装在 Netbeans 11 上吗?
- functional-programming - 重载函数的 Hindley-Milner 类型推断