django:具有多对多关系的values字典

如何解决django:具有多对多关系的values字典

我正试图(出于培训目的)申请图书馆目录。我决定为参与本书创作的所有人员创建一个Person模型,并通过中介模型将其与Book模型结合起来(因为一本书可能有数位作者,编校者,翻译者等),这将描述此人的角色

Book模型用于一本书的版本。它存储有关书籍的信息作为文学作品。每本书的每个版本都有一个图书实例。

复制模型是针对一本书的对象。

现在我正在尝试使搜索功能。我的目标是将符合搜索条件的所有书籍的列表显示为一行,其中包含与作者和书名相同的所有副本(无论是否是相同的出版商,翻译者等)。份数。因此,我了解,我应该在annotate字典上将Countvalues()一起使用。但是我无法获得使用values()时需要的所有信息。

如果我通过书名进行查询,则所有创建者都在values()字典中。但是,如果搜索条件是作者,则dictionnary值仅保留用于过滤的名称,因此,如果该书有两个或三个作者,我就不会知道他们的名字。

这是我的模型(对于Book模型,只有相关的字段,以使其更短):

class Person(models.Model):
    name = models.CharField(max_length=128,blank=True)
    born = models.CharField(max_length=64,blank=True)

class Book(models.Model):
    title = models.CharField(max_length=255,blank=True)
    creators = models.ManyToManyField(Person,through='Creator')
    
class Copy(models.Model):
    on_shelf = models.BooleanField()
    location = models.CharField(max_length=32,blank=True)
    remarques = models.CharField(max_length=255,blank=True)
    signature_mark = models.CharField(max_length=32,blank=True)
    approuved = models.BooleanField(default=False)
    collection = models.ForeignKey(Collection,on_delete=models.CASCADE,blank=True,null=True)
    book = models.ForeignKey(Book,null=True)

class Creator(models.Model):
    AUTHOR = 'A'
    TRANSLATOR = 'T'
    REDACTION = 'R'
    INTRODUCTION = 'I'
    ROLE_CHOICES = [
        (AUTHOR,'Autor'),(TRANSLATOR,'Tłumacz_ka'),(REDACTION,'Opracowanie,redakcja [Nazwisko Imię]'),(INTRODUCTION,'Wstęp,posłowie'),]
    
    person = models.ForeignKey(Person,on_delete=models.CASCADE)
    book = models.ForeignKey(Book,null=True)
    role = models.CharField(max_length=32,choices=ROLE_CHOICES)

这是我的疑问:

按标题:

book = Copy.objects.filter(book__title='Rozwój cywilizacji amerykańskiej tom 1-2').values('book__creators__name','book__creator__role','book__title') 

输出为:

{'book__creators__name': 'Garczyński Stefan','book__creator__role': 'T','book__title': 'Rozwój cywilizacji amerykańskiej tom 1-2'}
{'book__creators__name': 'Święcka Teresa','book__title': 'Rozwój cywilizacji amerykańskiej tom 1-2'}
{'book__creators__name': 'Beard Charles','book__creator__role': 'A','book__title': 'Rozwój cywilizacji amerykańskiej tom 1-2'}
{'book__creators__name': 'Beard Mary','book__title': 'Rozwój cywilizacji amerykańskiej tom 1-2'}

作者:

book = Copy.objects.filter(book__creators__name='Beard Mary',book__creator__role='A').values('book__creators__name','book__title')

输出:

{'book__creators__name': 'Beard Mary','book__title': 'Rozwój cywilizacji amerykańskiej tom 1-2'}

视图和模板只是草稿,尚无法使用。我首先尝试在外壳中找到解决方案。

视图:

class Searchg(View):
    def get(self,request,*args,**kwargs):
        form = SearchForm()
        
        return render(request,'working/search.html',{'form': form})

    def post(self,**kwargs):
        form = SearchForm(request.POST)
        if form.is_valid():
            author = form.cleaned_data['author']
            title = form.cleaned_data['title']
            if author and title:
                books = Book.objects.filter(creators__name__icontains=author,creator__role='A',title__icontains=title).order_by('title')
                print('a ',len(books))
            elif author:
                books = Book.objects.filter(creators__name__icontains=author,creator__role='A').order_by('title') 
                print('b ',len(books))
            elif title:
                books = Book.objects.filter(title__icontains=title).order_by('creators__name') 
                print('c ',len(books))
            else:
                print("Please enter search criteria")
            books = books.values('creators__name','title').annotate(title_count=Count('title'))
            paginator = Paginator(books,25)    
            page_number = request.GET.get('page')
            page_obj = paginator.get_page(page_number)
            cache.set('books',books)
        return redirect(reverse('working:book_list'))

和模板:

{% if page_obj %}

<table class="table table-bordered">
  <thead>
    <tr>
      <th scope="col"></th>
      <th scope="col"> Liczba egzemplarzy</th>
      <th scope="col">Tytuł</th>
      <th scope="col">Autor</th>
      
    </tr>
  </thead>
  <tbody>
    <tr>
        {% for book in page_obj %}
        
      <th scope="row">{{ forloop.counter0|add:page_obj.start_index}}</th>
      <td>{{ book.title_count }}</td>
      <td><em> {{ book.title }}</em></td>
      <td> {{ book.creators__name }}</td>   
      
           
    </tr>
    {% endfor %}
  </tbody>
</table>

<div class="pagination">
    <span class="step-links">
        {% if page_obj.has_previous %}
        <button class="btn btn-light"><a href="?page=1">&laquo; first</a></button>
        <button class="btn btn-light"><a href="?page={{ page_obj.previous_page_number }}">previous</a> </button>    
            
        {% endif %}

        <span class="current">
            Page {{ page_obj.number }} of {{ page_obj.paginator.num_pages }}.
        </span>

        {% if page_obj.has_next %}
        <button class="btn btn-light"><a href="?page={{ page_obj.next_page_number }}">next</a></button>
        <button class="btn btn-light"><a href="?page={{ page_obj.paginator.num_pages }}">last &raquo;</a></button>    
            
        {% endif %}

解决方法

这是使用.values(…) [Django-doc]常常不是一个好主意的众多原因之一。通常,人们使用它来分组一个特殊的字段,等等。

此外,您可能想在Book模型上查询,而不是在Copy模型上查询,因为这会使呈现每条记录的书变得更加困难。因此,您可以使用以下内容进行过滤:

from django.db.models import Count

books = Book.objects.filter(
    creator__name='Beard Mary',creator__role='A'
).annotate(
    ncopies=Count('copy')
).prefetch_related('creator_set')

,然后在模板中,您可以使用以下方式呈现该图片:

{% for book in books %}
    {{ book.title }}: {{ book.ncopies }} copies
    {% for creator in book.creator_set.all %}
        {{ creator.person.name }}: {{ creator.role }}
    {% endfor %}
{% endfor %}

这里.creator_set不会考虑对QuerySet的过滤,因为这是在单独的查询中处理的。

您还可以查询模板中的book.copy_set.all,以检索所有相关的Copy对象。

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


依赖报错 idea导入项目后依赖报错,解决方案:https://blog.csdn.net/weixin_42420249/article/details/81191861 依赖版本报错:更换其他版本 无法下载依赖可参考:https://blog.csdn.net/weixin_42628809/a
错误1:代码生成器依赖和mybatis依赖冲突 启动项目时报错如下 2021-12-03 13:33:33.927 ERROR 7228 [ main] o.s.b.d.LoggingFailureAnalysisReporter : *************************** APPL
错误1:gradle项目控制台输出为乱码 # 解决方案:https://blog.csdn.net/weixin_43501566/article/details/112482302 # 在gradle-wrapper.properties 添加以下内容 org.gradle.jvmargs=-Df
错误还原:在查询的过程中,传入的workType为0时,该条件不起作用 &lt;select id=&quot;xxx&quot;&gt; SELECT di.id, di.name, di.work_type, di.updated... &lt;where&gt; &lt;if test=&qu
报错如下,gcc版本太低 ^ server.c:5346:31: 错误:‘struct redisServer’没有名为‘server_cpulist’的成员 redisSetCpuAffinity(server.server_cpulist); ^ server.c: 在函数‘hasActiveC
解决方案1 1、改项目中.idea/workspace.xml配置文件,增加dynamic.classpath参数 2、搜索PropertiesComponent,添加如下 &lt;property name=&quot;dynamic.classpath&quot; value=&quot;tru
删除根组件app.vue中的默认代码后报错:Module Error (from ./node_modules/eslint-loader/index.js): 解决方案:关闭ESlint代码检测,在项目根目录创建vue.config.js,在文件中添加 module.exports = { lin
查看spark默认的python版本 [root@master day27]# pyspark /home/software/spark-2.3.4-bin-hadoop2.7/conf/spark-env.sh: line 2: /usr/local/hadoop/bin/hadoop: No s
使用本地python环境可以成功执行 import pandas as pd import matplotlib.pyplot as plt # 设置字体 plt.rcParams[&#39;font.sans-serif&#39;] = [&#39;SimHei&#39;] # 能正确显示负号 p
错误1:Request method ‘DELETE‘ not supported 错误还原:controller层有一个接口,访问该接口时报错:Request method ‘DELETE‘ not supported 错误原因:没有接收到前端传入的参数,修改为如下 参考 错误2:cannot r
错误1:启动docker镜像时报错:Error response from daemon: driver failed programming external connectivity on endpoint quirky_allen 解决方法:重启docker -&gt; systemctl r
错误1:private field ‘xxx‘ is never assigned 按Altʾnter快捷键,选择第2项 参考:https://blog.csdn.net/shi_hong_fei_hei/article/details/88814070 错误2:启动时报错,不能找到主启动类 #
报错如下,通过源不能下载,最后警告pip需升级版本 Requirement already satisfied: pip in c:\users\ychen\appdata\local\programs\python\python310\lib\site-packages (22.0.4) Coll
错误1:maven打包报错 错误还原:使用maven打包项目时报错如下 [ERROR] Failed to execute goal org.apache.maven.plugins:maven-resources-plugin:3.2.0:resources (default-resources)
错误1:服务调用时报错 服务消费者模块assess通过openFeign调用服务提供者模块hires 如下为服务提供者模块hires的控制层接口 @RestController @RequestMapping(&quot;/hires&quot;) public class FeignControl
错误1:运行项目后报如下错误 解决方案 报错2:Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:3.8.1:compile (default-compile) on project sb 解决方案:在pom.
参考 错误原因 过滤器或拦截器在生效时,redisTemplate还没有注入 解决方案:在注入容器时就生效 @Component //项目运行时就注入Spring容器 public class RedisBean { @Resource private RedisTemplate&lt;String
使用vite构建项目报错 C:\Users\ychen\work&gt;npm init @vitejs/app @vitejs/create-app is deprecated, use npm init vite instead C:\Users\ychen\AppData\Local\npm-