[Django-04]一对一,一对多

参数

OneToOneField(),ForeignKey() 的参数如下

  1. to 要连接的模型
  2. to_field 要被关联的目标Model的字段
  3. on_delete 删除主表(被关联的表,也就是非OneToOneField,ForeignKey的表)时,当前表怎么处理。
    CASCADE->联动删除
    PROTECT ->抛出异常
    SET_NULL->设为null
    SET_DEFAULT->设为预定义的默认值
    SET->设置为指定的值
    DO_NOTHIND ->什么也不做
class Grade(models.Model):
    id = models.AutoField(verbose_name='自增id', name="id", primary_key=True)
    grade_name = models.CharField(verbose_name="班级名称", name="grade_name", max_length=200)
    # grade_name = models.CharField(verbose_name="班级名称", db_column="grade_name", max_length=200)
    create_time = models.DateTimeField(verbose_name='创建时间',name="create_time", auto_now_add=True)  # 第一次赋值
    update_time = models.DateTimeField(verbose_name='最后一次更新时间',name="update_time", auto_now=True)  # 第二次赋值
    class Meta:
        db_table = "app_grade"
        ordering=["-update_time","create_time"]


class GradeManager(models.Model):
	"""
	一对一
	"""
    id = models.AutoField(verbose_name='自增id', name="id", primary_key=True)
    manager_name = models.CharField(verbose_name="班主任名称", db_column="manager_name", max_length=200,default=None)
    create_time = models.DateTimeField(verbose_name='创建时间',db_column="create_time", auto_now_add=True)  # 第一次赋值
    update_time = models.DateTimeField(verbose_name='最后一次更新时间',db_column="update_time", auto_now=True)  # 第二次赋值
    # 下面会在数据库中生成grade_id字段
    grade=models.OneToOneField(verbose_name='班主任表', to=Grade,on_delete=models.CASCADE)
    class Meta:
        db_table = "app_grade_manager"
        ordering=["-update_time","create_time"]

class Student(models.Model):
    """
    一对多
    to: 关联的父 类
    to_filed:
    外键要放在 多表中
    删除父表数据时,当前表与关联表的filed的行为。
    ForeignKey->字段名称会自动拼接id, 因此对应的序列化对象要加上id
    """
    id = models.AutoField(verbose_name='自增id', name="id", primary_key=True)
    name = models.CharField(verbose_name='名字', name='name', max_length=200)
    address = models.JSONField(verbose_name="地址信息", name="address")
    create_time = models.DateTimeField(verbose_name='创建时间', name='create_time', auto_now_add=True)  # 第一次赋值
    update_time = models.DateTimeField(verbose_name='最后一次更新时间', name='update_time', auto_now=True)  # 第二次赋值
    # 下面会在数据库生成grade_id字段
    grade = models.ForeignKey(verbose_name="关联外键", name="grade", to=Grade, on_delete=models.CASCADE)

    class Meta:
        db_table = "app_student"
        ordering=["-update_time","create_time"]

一对一 查询

grade = Grade.objects.get(id=3)
print(grade.grademanager.manager_name) # 反向通过model小写名称引用
gradeManager = GradeManager.objects.get(id=1)
print(gradeManager.grade.grade_name) # 正向直接通过属性grade引用

一对多 查询

**************************************** 一,找到 “多” ************************************************

grade = Grade.objects.get(id=3)
for student in grade.student_set.all():
    print(student.name)

*************************************** 多,找到"" ************************************************

s = Student.objects.()
# 这种写法(s.grade)会查一次数据库
	print(s.grade.grade_name)
	
下面是优化的写法:
s = Student.objects.select_related("grade").all() #这种写法会用inner join
	print(s.grade.grade_name)
意思是,将id=3的student拿到,同事将grade用 innor join同步查出来
Django中,处理一对多(Many-to-Many)关系的新建和更新通常涉及到两个模型之间的关联表。例如,如果你有两个模型`Author`和`Book`,作者可以有多本书,书也可以有多个作者,这就是一对多关系。以下是使用`QuerySet`在Django中进行操作的一般步骤: 1. **新建数据**: - 要添加一个新的作者及其关联的书籍,你可以先创建作者实例,然后通过属性访问其`books`关联集合: ```python author = Author.objects.create(name="John Doe") book = Book(title="Book Title") # 将新书添加到作者的书籍列表 author.books.add(book) ``` - 或者,如果`Book`有一个反向引用(`author_set`): ```python book.authors.add(author) ``` 2. **更新数据**: - 如果你想更新一个作者已有的书籍,例如删除一本书或替换书籍列表,可以使用`remove()`或`clear()`方法: ```python # 删除某本书 author.books.remove(book) # 清空所有书籍 author.books.clear() # 或者添加新的书籍 author.books.add(Book(title="New Book")) ``` 3. **通过外键更新关联**: - 如果你想要更新书籍信息,并自动更新所有关联的作者,可以使用`save()`方法,因为它会触发数据库的级联更新规则: ```python book.title = "Renamed" book.save() # 这将会更新所有与book相关的author对象的title字段 ``` 记得在实际操作时,检查并处理可能出现的异常,比如试图删除最后一本书可能会导致错误。同时,确保你理解了你的数据模型结构以及它们之间的约束。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

我先森

鼓励一个吧,哈哈

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值