使用Django实现把两个模型类的数据聚合在一起

作者:肾虚少年 时间:2023-11-11 23:37:42 

Django中想要把模型类聚合得到想要的数据可以用F对象。

比如有模型类A和B,A和B之间有外键关联在一起,A是子表,B是父表(反过来没试过。。因为大部分数据都是用子表的,我想是可以的),那么可以这样查:

A.objects.filter(userid=3,bookid=F(bid))

其中userid,bookid是模型类A的字段,bid是模型类B的字段。

这样操作的结果就是可以查询到userid为3且模型类A字段bookid等于模型类B字段bid的集合数据了。

F对象是可以比较两个关联模型类的字段数据的。

我看到网上有说F对象可以这样用F('b__id') ==>F('模型类名小写__字段名'),此处是双下划线。

但是我用Django2.0时会报错。。真是搞不懂,后来我直接使用字段名居然可以,醉了。

反正都可以试试吧

A.objects.filter(userid=3,bookid=F(bid))

A.objects.filter(userid=3,bookid=F('b__bid'))

补充知识:Django Admin页面显示父表,编辑子表

默认情况下,ModelAdmin只允许您管理模型“本身”字段,而不是相关模型.以下方法将实现,在应用类的列表管理显示页面,显示父表的字段;在编辑页面,父表对子表进行编辑.

models.py如下


class Level(models.Model):
# l_num = models.IntegerField(default=0, verbose_name='序号')
name = models.CharField(max_length=20)
is_delete = models.BooleanField(default=False)

def __str__(self):
 return self.name

class Grades(models.Model):
name = models.CharField(max_length=20, verbose_name='班级')
# 外键,这里关联模型Level与模型名称一样,不是全小写。
level = models.ForeignKey('Level', on_delete=models.DO_NOTHING)
is_delete = models.BooleanField(default=False)

def __str__(self):
 return '%s%s' % (self.level, self.name)

class Students(models.Model):
name = models.CharField(max_length=20, db_index=True, verbose_name='姓名')
age = models.IntegerField(verbose_name='年龄')

# 需要先提供一个二维的二元元组,第一个元素表示存在数据库内真实的值,第二个表示页面上显示的具体内容
SEX_CHOICE = (
 ('男', '男'),
 ('女', '女'),
)
sex = models.CharField(max_length=10, choices=SEX_CHOICE, verbose_name='性别', default='男')
grade = models.ForeignKey('Grades', on_delete=models.DO_NOTHING, verbose_name="班级")
img_student = models.ImageField(upload_to='img_student', default='img_student/default.png', verbose_name='头像')
create_time = models.DateTimeField(auto_now_add=True, verbose_name='创建时间')
last_update_time = models.DateTimeField(auto_now=True, verbose_name='上次更新时间')
is_delete = models.BooleanField(default=False)

def __str__(self):
 return self.name

在admin.py代码如下:


@admin.register(Students)
class StudentsAdmin(admin.ModelAdmin):
list_display = ('name', 'age', 'sex', 'grade', 'create_time', 'last_update_time', 'is_delete')

做完之后,显示的效果如下:

使用Django实现把两个模型类的数据聚合在一起

在应用类的列表管理显示页面,显示父表的字段

可以让Students,显示父表Grades的父表Level字段

在models.py里的Students类里,写上如下代码:

class Students(models.Model):


 # 写一个方法,定义在管理页面上能够显示的外键字段字段
 # grade为Students模型的外检表,level为Grades模型的外检表,那么为Level模型的字段
 def dis_level(self):
  return self.grade.level.name

# 定义该字段在管理后台显示的名称
 dis_level.short_description = '年级'

# 定义该字段在管理后台显示的名称
dis_level.short_description = '年级'
# 方法列是不能排序的,如果需要排序需要为方法指定排序依据。添加的是'模型类字段'
# 如果是外键需要遵循这样的语法:本表外键字段__(双下划线)外检表字段或外检表的外键字段__最终外键表要显示的字段。
dis_level.admin_order_field = 'grade__level__name'

在admin.py里,把Students类里的方法,加入到list_display里:


@admin.register(Students)
class StudentsAdmin(admin.ModelAdmin):
list_display = ('name', 'age', 'sex', 'grade', 'dis_level', 'create_time', 'last_update_time', 'is_delete')

写完之后,显示的结果如下,多了年级,以及点击年级可以进行排序:

使用Django实现把两个模型类的数据聚合在一起

在编辑页面,父表对子表进行编辑.

默认对学生编辑时,无法在编辑页面直接编辑相关联的子表,例如:

models.py代码:


class Students(models.Model):
name = models.CharField(max_length=20, db_index=True, verbose_name='姓名')
age = models.IntegerField(verbose_name='年龄')

# 需要先提供一个二维的二元元组,第一个元素表示存在数据库内真实的值,第二个表示页面上显示的具体内容
SEX_CHOICE = (
 ('男', '男'),
 ('女', '女'),
)
sex = models.CharField(max_length=10, choices=SEX_CHOICE, verbose_name='性别', default='男')
grade = models.ForeignKey('Grades', on_delete=models.DO_NOTHING, verbose_name="班级")
img_student = models.ImageField(upload_to='img_student', default='img_student/default.png', verbose_name='头像')
create_time = models.DateTimeField(auto_now_add=True, verbose_name='创建时间')
last_update_time = models.DateTimeField(auto_now=True, verbose_name='上次更新时间')
is_delete = models.BooleanField(default=False)

def __str__(self):
 return self.name

class Course(models.Model):
name = models.CharField(max_length=20, verbose_name='课名')
is_delete = models.BooleanField(default=False)

def __str__(self):
 return self.name

class Score(models.Model):
s_score = models.IntegerField(default=0, verbose_name='分数')
s_course = models.ForeignKey('Course', on_delete=models.DO_NOTHING, verbose_name='课程')
s_student = models.ForeignKey('Students', on_delete=models.DO_NOTHING, verbose_name='学生姓名')
is_delete = models.BooleanField(default=False)

def __str__(self):
 # 要把s_score转换为字符串,否则会报下面的错误。
 """
  Exception Type:TypeError
  Exception Value:
  __str__ returned non-string (type int)
  :return:
 """
 return '%s%s%s' % (self.s_student, self.s_course, str(self.s_score))

打开学生的编辑页面,是这样子的:

使用Django实现把两个模型类的数据聚合在一起

要给学生添加分数,只能进入Score管理页面,一个个添加,非常麻烦.

使用Django的TabularInline,可以解决这个问题,在父表里对子表进行编辑:

所有代码都在admin.py里写,具体如下:


# 一对多关联表编辑,让父表管理配置页面能同时编辑子表,以下的Score为子表(有外键所在的表)
class ScoreInline(admin.TabularInline):
# Score 必须是models.py中的模型名称,大小写必须要匹配.这个模型为子表,以便可以被父表编辑
model = Score
# 默认显示条目的数量
# extra = 5

class StudentsAdmin(admin.ModelAdmin):
# Inline把ScoreInline关联进来,让父表管理配置页面能同时编辑子表.
inlines = [ScoreInline, ]

做完之后,效果如下:

使用Django实现把两个模型类的数据聚合在一起

来源:https://blog.csdn.net/woshidamimi0/article/details/79795462

标签:Django,模型类,数据,聚合
0
投稿

猜你喜欢

  • Yahoo! BrowserPlus 介绍

    2008-06-01 16:38:00
  • Pygame实战练习之一百层游戏

    2022-01-08 00:07:54
  • mysql 8.0.18 压缩包安装及忘记密码重置所遇到的坑

    2024-01-28 18:01:35
  • Django使用Channels实现WebSocket的方法

    2023-12-10 16:20:05
  • PYQT5 vscode联合操作qtdesigner的方法

    2023-10-31 16:46:41
  • python实现textrank关键词提取

    2021-11-01 16:47:58
  • t-sql清空表数据的两种方式示例(truncate and delete)

    2024-01-20 09:46:24
  • php中json_decode()和json_encode()的使用方法

    2023-06-11 16:31:56
  • 在Python的Django框架上部署ORM库的教程

    2021-04-08 02:20:47
  • python/Matplotlib绘制复变函数图像教程

    2023-08-03 07:36:43
  • python中pandas对多列进行分组统计的实现

    2022-06-25 03:50:03
  • 人工智能学习pyTorch自建数据集及可视化结果实现过程

    2022-08-04 14:54:33
  • MySQL中Innodb的事务隔离级别和锁的关系的讲解教程

    2024-01-13 18:20:21
  • pytorch 实现cross entropy损失函数计算方式

    2022-03-18 00:45:50
  • Python编程编写完善的命令行工具

    2023-08-02 11:22:56
  • Python使用Chrome插件实现爬虫过程图解

    2023-08-01 02:02:29
  • Vue双向绑定原理及实现方法

    2024-05-10 14:16:55
  • python脚本使用阿里云slb对恶意攻击进行封堵的实现

    2021-11-20 16:32:07
  • 如何用Python和JS实现的Web SSH工具

    2021-04-23 13:50:13
  • Python arrow 更好的日期时间模块

    2022-02-08 14:06:15
  • asp之家 网络编程 m.aspxhome.com