首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >问答首页 >如何加入这些Django模型?

如何加入这些Django模型?
EN

Stack Overflow用户
提问于 2019-05-13 13:37:41
回答 1查看 86关注 0票数 0

我应该如何加入这些Django模型,以获得所有参与登录用户挑战的玩家模型(请参阅views.py中名为challenges_participants的变量?关于如何使用Django模型有效地实现这一点,我没有想出一个解决方案。非常感谢!

示例数据:假设John已登录。正如我们在PlayerChallenge模型中看到的,他是挑战1和挑战2的一部分。我要做的是获取挑战1中的所有玩家对象(本例中是罗杰),以及挑战2中所有玩家对象的第二个列表(本例中是罗杰和克劳德)。

播放器模型

代码语言:javascript
运行
复制
╔═══════════╦═════════╦════════════════╗
║ user_name ║ user_id ║ wallet_balance ║
╠═══════════╬═════════╬════════════════╣
║ John      ║     123 ║             20 ║
║ Roger     ║     231 ║             15 ║
║ Claude    ║     582 ║             10 ║
╚═══════════╩═════════╩════════════════╝

挑战模型

代码语言:javascript
运行
复制
╔════╦═════════╦══════╦════════╦═══════════════════╦══════════╦═══════════════════╦═══════════════╗
║ id ║ status  ║ type ║ amount ║ created_timestamp ║ duration ║ started_timestamp ║ end_timestamp ║
╠════╬═════════╬══════╬════════╬═══════════════════╬══════════╬═══════════════════╬═══════════════╣
║  1 ║ active  ║    1 ║     20 ║ TimeStamp         ║ 02:00:00 ║ TimeStamp         ║ TimeStamp     ║
║  2 ║ waiting ║    2 ║     30 ║ TimeStamp         ║ 04:00:00 ║ TimeStamp         ║ TimeStamp     ║
║  3 ║ waiting ║    1 ║     35 ║ TimeStamp         ║ 07:00:00 ║ TimeStamp         ║ TimeStamp     ║
╚════╩═════════╩══════╩════════╩═══════════════════╩══════════╩═══════════════════╩═══════════════╝

PlayerChallenge模型

代码语言:javascript
运行
复制
╔════╦════════╦═══════════╦════════════════════╗
║ id ║ player ║ challenge ║ accepted_timestamp ║
╠════╬════════╬═══════════╬════════════════════╣
║  1 ║ John   ║         1 ║ TimeStamp          ║
║  2 ║ Roger  ║         1 ║ TimeStamp          ║
║  3 ║ Roger  ║         2 ║ TimeStamp          ║
║  4 ║ Claude ║         2 ║ TimeStamp          ║
║  5 ║ John   ║         2 ║ TimeStamp          ║
╚════╩════════╩═══════════╩════════════════════╝

代码:

models.py

代码语言:javascript
运行
复制
from django.db import models
from django.contrib.auth.models import User
from django.db.models.signals import post_save
from django.dispatch import receiver

# Create your models here.

class Player(models.Model):
    user = models.OneToOneField(User, on_delete=models.CASCADE)
    #additional fields
    userid = models.CharField(max_length=256)
    wallet_balance = models.IntegerField(default=0)
    def __str__(self):
        return self.user.username

@receiver(post_save, sender=User)
def create_user_profile(sender, instance, created, **kwargs):
    if created:
        Player.objects.create(user=instance)

@receiver(post_save, sender=User)
def save_user_profile(sender, instance, **kwargs):
    instance.player.save()


class Challenge(models.Model):
    id = models.IntegerField(primary_key=True, unique=True)
    status = models.CharField(max_length=20)
    type = models.CharField(max_length=20)
    amount = models.DecimalField(max_digits=10, decimal_places=2)
    created_timestamp = models.DateTimeField()
    duration = models.DurationField()
    started_timestamp = models.DateTimeField(null=True, blank=True)
    end_timestamp = models.DateTimeField(null=True, blank=True)

    def __str__(self):
        return str(self.id)

class PlayerChallenge(models.Model):
    user_challenge_id = models.IntegerField(primary_key=True, unique=True)
    player = models.ForeignKey(Player, on_delete=models.CASCADE)
    challenge = models.ForeignKey(Challenge, on_delete=models.CASCADE)
    accepted_timestamp = models.DateTimeField(null=True, blank=True)

    def __str__(self):
        return str(self.user_challenge_id) + " (Player: " + str(self.player.user.username) + ", Challenge: " + str(self.challenge.id) + ")"

views.py

代码语言:javascript
运行
复制
from django.shortcuts import render
from django.http import HttpResponse

from django.contrib.auth import authenticate, login, logout
from django.http import HttpResponseRedirect, HttpResponse
from django.urls import reverse
from django.contrib.auth.decorators import login_required
from django.contrib.auth.models import User

from challengeview.models import Player,Challenge,PlayerChallenge

@login_required
def index(request):

    challenges = Challenge.objects.filter(playerchallenge__player = request.user.player)
    challenges_participants = Player.objects.filter(playerchallenge__challenge = challenges)

    my_dict = {
     'challenges_participants' : challenges_participants,
     'challenges' : challenges,

    }
    return render(request, "challengeview/index.html", context=my_dict)

非常感谢您的支持!

EN

回答 1

Stack Overflow用户

发布于 2019-05-13 17:27:51

在Django中,您不能像在简单的SQL表中那样对关系进行建模。有三种强大的关系类型OnetoOne, ForeignKey, ManytoMany。从django.utils导入时区

代码语言:javascript
运行
复制
class Player(models.Model):
    user = models.OneToOneField(User, on_delete=models.CASCADE)
    #additional fields
    userid = models.CharField(max_length=256)
    wallet_balance = models.IntegerField(default=0)
    challenges = models.ManyToMany('Challenge') #a challenge can be belonged to many players in this case
    def __str__(self):
        return self.user.username

@receiver(post_save, sender=User)
def create_user_profile(sender, instance, created, **kwargs):
    if created:
        Player.objects.create(user=instance)

@receiver(post_save, sender=User)
def save_user_profile(sender, instance, **kwargs):
    instance.player.save()


class Challenge(models.Model):
    status = models.CharField(max_length=20)
    type = models.CharField(max_length=20)
    amount = models.DecimalField(max_digits=10, decimal_places=2)
    duration = models.DurationField()
    started_timestamp = models.DateTimeField(null=True, blank=True)
    end_timestamp = models.DateTimeField(null=True, blank=True)
    created_timestamp = models.DateTimeField()

    def __str__(self):
        return str(self.id)


class PlayerActivity(models.Model):
    challenge = models.ForeingKey(on_delete=models.SET_NULL,
                                   null=True,
                                 related_name='activity_challenge')
    created_by = models.ForeignKey(Player,
                                   on_delete=models.SET_NULL,
                                   null=True,
                                 related_name='activity_created_by')

    created_at = models.DateTimeField(default=timezone.now)

    def __str__(self):
        return str(self.user_challenge_id) + " (Player: " + str(self.player.user.username) + ", Challenge: " + str(self.challenge.id) + ")"

如果一个挑战只能属于一个玩家,但是一个玩家可以有许多挑战,那么Foreignkey。如果一个玩家可以有多个挑战,一个挑战可以属于多个玩家,那么ManyToMany

当您使用查询时,您应该始终考虑发送到后端的请求。您可以使用select_related最小化请求。

代码语言:javascript
运行
复制
all_challenges = Challenge.objects.all()
logged_user_challenges = Player.objects.filter(user=request.user,challenges__in=all_challenges)
others_same_challenges = Player.objects.filter(challenges=logged_user_challenges.challenges)

代码语言:javascript
运行
复制
logged_user_challenges = PlayerActivity.objects.filter(user=request.user).latest('created_at')

但是你会有一个others_的问题...你需要一个循环来得到所有的东西,这是不被建议的。

注意:您必须在视图中操作数据保存。

票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/56106289

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档