Django ORM 无法通过 `ForeignKey` 自动关联,而是需要 **根据父模型中的某个字段(比如 ID)去查询子模型**。

Django ORM 无法通过 ForeignKey 自动关联,而是需要 根据父模型中的某个字段(比如 ID)去查询子模型

在 DRF 里,这种情况可以通过 SerializerMethodField自定义字段 来实现。


示例场景

假设:

python 复制代码
class Parent(models.Model):
    id = models.AutoField(primary_key=True)
    name = models.CharField(max_length=100)
    # 没有外键,只有 parent_id 用来查询子模型

class Child(models.Model):
    id = models.AutoField(primary_key=True)
    parent_id = models.IntegerField()  # 用 parent.id 关联
    content = models.CharField(max_length=200)

1️⃣ 使用 SerializerMethodField 获取子模型

python 复制代码
from rest_framework import serializers
from .models import Parent, Child

class ChildSerializer(serializers.ModelSerializer):
    class Meta:
        model = Child
        fields = ("id", "content")

class ParentSerializer(serializers.ModelSerializer):
    children = serializers.SerializerMethodField()

    class Meta:
        model = Parent
        fields = ("id", "name", "children")

    def get_children(self, obj):
        # 根据父模型的 id 查询子模型
        qs = Child.objects.filter(parent_id=obj.id)
        return ChildSerializer(qs, many=True).data

2️⃣ 在 View 中使用

python 复制代码
from rest_framework import generics
from .models import Parent
from .serializers import ParentSerializer

class ParentListAPIView(generics.ListAPIView):
    queryset = Parent.objects.all()
    serializer_class = ParentSerializer

class ParentDetailAPIView(generics.RetrieveAPIView):
    queryset = Parent.objects.all()
    serializer_class = ParentSerializer

3️⃣ 优化性能

如果父模型很多,直接在 get_children 查询会导致 N+1 查询,可以提前把子模型查出来,然后传给 serializer:

python 复制代码
class ParentListAPIView(generics.ListAPIView):
    serializer_class = ParentSerializer

    def get_queryset(self):
        return Parent.objects.all()

    def list(self, request, *args, **kwargs):
        parents = self.get_queryset()
        parent_ids = [p.id for p in parents]
        children_qs = Child.objects.filter(parent_id__in=parent_ids)

        # 按 parent_id 分组
        children_map = {}
        for c in children_qs:
            children_map.setdefault(c.parent_id, []).append(c)

        # 把 children_map 传给 serializer context
        serializer = self.get_serializer(parents, many=True, context={"children_map": children_map})
        return Response(serializer.data)

然后在 serializer 里:

python 复制代码
def get_children(self, obj):
    children_map = self.context.get("children_map", {})
    qs = children_map.get(obj.id, [])
    return ChildSerializer(qs, many=True).data

这样只会做 两条 SQL 查询,避免 N+1。


✅ 总结

  • 没有外键关系 → 用 SerializerMethodField 或自定义字段查询
  • N+1 查询问题 → 可以提前批量查询子模型并用 context 传递
  • 适用于一对多、多对多但没有数据库约束的场景

如果你需要,我可以帮你写一个 通用方法,输入父模型 queryset 和子模型 queryset,自动生成嵌套返回的数据,完全不依赖外键。你希望我直接写吗?

相关推荐
不会代码的小测试2 分钟前
UI自动化-POM封装
开发语言·python·selenium·自动化
2401_841495644 分钟前
【LeetCode刷题】二叉树的层序遍历
数据结构·python·算法·leetcode·二叉树··队列
毕设源码-钟学长10 分钟前
【开题答辩全过程】以 基于Springboot的扶贫众筹平台为例,包含答辩的问题和答案
java·spring boot·后端
ZH154558913114 分钟前
Flutter for OpenHarmony Python学习助手实战:GUI桌面应用开发的实现
python·学习·flutter
B站计算机毕业设计超人19 分钟前
计算机毕业设计Hadoop+Spark+Hive招聘推荐系统 招聘大数据分析 大数据毕业设计(源码+文档+PPT+ 讲解)
大数据·hive·hadoop·python·spark·毕业设计·课程设计
B站计算机毕业设计超人20 分钟前
计算机毕业设计hadoop+spark+hive交通拥堵预测 交通流量预测 智慧城市交通大数据 交通客流量分析(源码+LW文档+PPT+讲解视频)
大数据·hive·hadoop·python·spark·毕业设计·课程设计
CodeSheep程序羊25 分钟前
拼多多春节加班工资曝光,没几个敢给这个数的。
java·c语言·开发语言·c++·python·程序人生·职场和发展
独好紫罗兰26 分钟前
对python的再认识-基于数据结构进行-a002-列表-列表推导式
开发语言·数据结构·python
机器学习之心HML28 分钟前
多光伏电站功率预测新思路:当GCN遇见LSTM,解锁时空预测密码,python代码
人工智能·python·lstm
2401_8414956430 分钟前
【LeetCode刷题】二叉树的直径
数据结构·python·算法·leetcode·二叉树··递归