使用ORM组合数量的数据库请求

时间:2014-02-17 11:04:48

标签: django django-models django-orm

我正在研究website,用户可以在其中对不同类别的不同项目进行投票和评论。每当用户返回该站点时,他们的投票将被突出显示并且所有评论都将可见。 问题是,每次加载页面时,由于我的模型设计或由于我没有找到有效的方法来检索数据,因此会有许多数据库调用。

我的模型如下:

class ProjectCategory:
    title=models.CharField(...)

class Project:
    category=models.ForeignKey(ProjectCategory)
    title=models.CharField(...)

    def ownedComments(self):
            return Comment.objects.filter(project=self).order_by('-submissionTime')

    def ownedCommentsPreview(self):
            return Comment.objects.filter(project=self).order_by('-submissionTime')[0:3]

    def ownedVotes(self):
            return Vote.objects.filter(project=self,upVoted=True).count() 


class Vote:
    project=models.ForeignKey(Project)
    user=models.ForeignKey(User)
    vote=models.IntegerField(...)

class Comment:
    project=models.ForeignKey(Project)
    user=models.ForeignKey(User)
    title=models.CharField(...)    

以下是 views.py 的相关部分:

def getCurrentUserData(request=None):
        if (request==None) or (request.user.is_authenticated()==False):
                return {'loggedInUsername':'Anonym','userIsLoggedIn':False}
        return {'loggedInUsername':request.user.username,'userIsLoggedIn':True}


def getProjectResponse(request,surveyFile=None,additionalInfo={}):
        userVotedUp=[]
        if surveyFile!=None:
                userVotes=Vote.objects.filter(upVoted=True,surveyFile=surveyFile)
                for vote in userVotes:
                        userVotedUp.append(vote.project.id)
        answerDict={'userVotedUp':userVotedUp,'prjCategories':PrjCategory.objects.select_related(),'projects': Project.objects.prefetch_related('category').filter(visible=True),'spammerquestion':spammerquestion.SpammerQuestion().getRandomQuestion(),'rotatingLines':helpers.getRandomRotatingLines(),'backgroundimageurl':helpers.getBackgroundImageUrl()}
        answerDict.update(getCurrentUserData(request))
        answerDict.update(additionalInfo)
        return answerDict

def showProject(request,prjNo):
        currentPrj=get_object_or_404(Project,prjNo=prjNo,visible=True)
        surveyFile=assignSurveyFile(request,create_surveyFile_if_not_existing=False)        
        return render_to_response('website/singleproject.html', getProjectResponse(request,surveyFile,{'currentPrj':currentPrj,'immediateProjectID':currentPrj.id}),context_instance=RequestContext(request))

模板包含许多项目和类别的迭代,这里可能会发生多个数据库调用。 模板的相关部分是:

{% for project in projects %}
    self.vote['{{ project.id }}']=ko.observable(new Vote({'totalVoteNumber':ko.observable({{ project.ownedVotes }}),'userVoteState':ko.observable({% if project.id in userVotedUp %}true{% else %}false{% endif %})}));
    self.comments['{{ project.id }}']=ko.observableArray([]);
    {% for comment in project.ownedCommentsPreview %}
        self.comments['{{ project.id }}'].push(new Comment({'description':'{{ comment.description|striptags|safe }}','submissionTime':'{{ comment.submissionTime|humanDT }}','commentUuid':'{{ comment.commentUuid }}','deletable':false}));
    {% endfor %}
{% endfor %}


        {% for prjCat in prjCategories %}
                .prjCatUnderline{{ prjCat.id }} {border-bottom: 0px solid {{ prjCat.color }};}
                .prjCatUnderline{{ prjCat.id }}:hover {border-bottom: 3px solid {{ prjCat.color }};}
        {% endfor %}


        {% for prjCat in prjCategories %}
                <a href="#" class="prjCatUnderline{{ prjCat.id }}" data-filter=".prjCat{{ prjCat.id }}">{{ prjCat.title }}</a>
                {% if not forloop.last %}<span class="catpipe">|</span>{% endif %}
        {% endfor %}

        <script type="text/javascript">
                var projectLocations = [{% for project in projects %}{% if project.location != '' %}['{{ project.title }}', {{ project.location }}, {{ project.id }}, '{{ MEDIA_URL }}{{ project.mapImage }}', {{ project.mapImage.width }}, {{ project.mapImage.height }}, '{{ project.id }}'],
                {% endif %}{% endfor %}];
        </script>


        {% for project in projects %}
                <div class="item prjCat{{ project.category.id }}">
                        <div class="prjnumberbox">
                                <div class="numberbar" style="background-color:{{ project.category.color }}"></div>
                                <div class="prjnumber">{{ project.prjNo|stringformat:"02d" }}</div>
                                <div class="numberbar" style="background-color: {{ project.category.color }}"></div>
                        </div>
                        <div class="prjheadline"><span>{{ project.title }}</span></div>
                        <div class="prjimagecontainer">
                                <!-- ko with: $root.vote['{{ project.id }}'] -->
                                <div class="prjimageoverlay"><a data-bind="click: function(data,event) { $root.showSingleProjectData('{{ project.id }}',data) },css : { imgVoted :userVoteState }" class="singleProjectOverlay" href="{% url 'website.views.showProject' prjNo=project.prjNo %}"></a></div>
                                <div class="prjimage"><img src="{{ MEDIA_URL }}{{ project.teaserImage }}" /></div>
                                <!-- /ko -->
                        </div>
                        <div class="prjtextcontainer" {% if project.ownedComments|length_is:"0" %}style="background-image:none;"{% endif %}>
                                <div class="prjtext">{{ project.teaserText|safe }} <a class="singleProjectOverlay" data-bind="click: function(data,event) { showSingleProjectData('{{ project.id }}',data) }" href="{% url 'website.views.showProject' prjNo=project.prjNo %}">mehr &gt;&gt;</a></div>
                                <div class="votesline"><span class="voteslinefont" data-bind="with: vote['{{ project.id }}']">&nbsp;<span class="votecount" data-bind="text: totalVoteNumber">{{ project.ownedVotes }}</span> votes&nbsp;</span></div>

                                <div class="prjcomments nojs">
                                {% for comment in project.ownedCommentsPreview|slice:":4" %}
                                        <div class="prjcommentitem">
                                                {{ comment.teaserdescription }}
                                        </div>
                                {% endfor %}
                                </div>
                                <div class="prjcomments" data-bind="foreach: comments['{{ project.id }}'], visible: comments['{{ project.id }}']().length > 0">
                                        <!-- ko if: $index() < 4 -->
                                        <div class="prjcommentitem">
                                                <em data-bind="text: submissionTime">&nbsp;</em><br/>
                                                <span data-bind="text: description">&nbsp;</span>
                                                <a href="#" data-bind="click: function(data, event) { $parent.removeComment(data,'{{ project.id }}', data,event) }, visible: deletable">[X]</a>
                                        </div>
                                        <!-- /ko -->

                                </div>
                        </div>
                </div>
        {% endfor %}

对于首页,我想检索所有项目,最近的三条评论,项目的总投票数以及用户的评论和投票。 如果可以通过一次数据库调用,或者两次,一般用于获得投票数和评论数,以及一项用于特定于用户的数据,那将是很好的。

我正在通过一次调用检索所有项目,因此我自动获取所有项目类别。然后我立刻得到所有的评论并将它们分配给python中的项目。然后我得到某个用户的评论和投票,每个用户都有一个电话。

有没有办法 - 不直接使用sql - 来获取

  • 对具有单个数据库调用的用户的评论和投票
  • 评论,项目,每个项目的累计投票数和项目类别一次?

一个想法是使用select_related()下载整个数据库,然后在Python中对所有内容进行排序,但如果将来我有大量的投票或评论,这可能不是一个好主意。

0 个答案:

没有答案