获取一些包含多对多外键但是不包含特定另一个外键的结果

前天碰到了一个情况,思考了好久没能想出问题,昨天下午在 stackoverflow 上问了一下,有一个好心人跟我沟通了一下,了解清楚我的问题以后,到了晚上给出了答案,我操作了一下果然是正确的,让我感动不已。他是一个好人,好人一生平安。

问题

开始描述我的问题吧,标题不能描述清楚我的问题,我应该在正文详细描述一下

用户( User )发布状态( Feed )评论( Comment ) 三个 Model,分别定义如下

class User(models.Model):
    name = models.CharField(max_length=20)


class Feed(models.Model):
    user = models.ForeignKey(User, related_name="feed_user")


class Comment(models.Model):
    user = models.ForeignKey(User, related_name="comment_user")
    feed = models.ForeignKey(Feed, related_name="comment_feed")

数据库目前的记录

User

| id | name |
|---|---|---|
| 1 | Jack |
| 2 | Bruce |

Feed

id user
1 1

Comment

id user feed
1 2 1
2 1 1

我需要获取一个用户的所有 Feed ,每一个 Feed 包含它所有的 Comment ,但是每一个 Comment 对应的 user 不能是 Feed 对应的 user 。

说白了我就是想获取不是我自己的所有 Comment 。

解决方法

我能想到的就是使用 exclude 、 Q 这些方法。所以一开始我的获取方式如下:

获取 Jack 的所有 Feed

从上面的数据库记录来看, Jack 发了一条 Feed, 这条 Feed 有两条 Comment,一条是 Bruce 评论的,还有一条是 Jack 自己评论的。

Feed.objects.filter(user=1)

序列化结果如下

[
    {
        "feed": {
            "id": 1,
            "user": {
                "id": 1
            },
            "comment": [
                {
                    "id": 1,
                    "user": {
                        "id": 2
                    },
                    "feed": {
                        "id": 1,
                        "user": {
                            "id": 1
                        }
                    }
                },
                {
                    "id": 2,
                    "user": {
                        "id": 1
                    },
                    "feed": {
                        "id": 1,
                        "user": {
                            "id": 1
                        }
                    }
                }
            ]
        }
    }
]

使用 exclude 方法过滤 Jack 自己评论的结果

Feed.objects.filter(user=1).exclude(feed_user__user=1)

这种方法获取的序列化结果

[
    {
        "feed": {
            "id": 1,
            "user": {
                "id": 1
            },
            "comment": []
        }
    }
]

此时的序列化结果竟然把整个 comment 过滤掉了

通过查看 queryset 的 SQL 语句

SELECT `feed_feed`.`id`, `feed_feed`.`user_id` FROM `feed_feed` WHERE NOT (`feed_feed`.`id` IN (SELECT U1.`feed_id_id` AS Col1 FROM `feed_comment` U1 WHERE U1.`user_id` = 1)) ORDER BY `feed_feed`.`create_time` DESC

发现它是通过判断 Jack 是否出现在所有评论中的某一个, 因此如果出现了就认为整个都不要了,这种方式不可取。

prefetch_related 返回一个 QuerySet ,为每个关系单独查找,并在 Python 中“加入”。这允许它预取多对多和多对一对象,除了外键和一对一关系。

Feed.objects.filter(user=1).prefetch_related(
    Prefetch(
        "feed_comment",
        queryset=Comment.objects.exclude(
            user=1
        ),
        to_attr="all_comment"
    )
)

此时的序列化结果如下

[
    {
        "feed": {
            "id": 1,
            "user": {
                "id": 1
            },
            "comment": [
                {
                    "id": 1,
                    "user": {
                        "id": 2
                    },
                    "feed": {
                        "id": 1,
                        "user": {
                            "id": 1
                        }
                    }
                },
                {
                    "id": 2,
                    "user": {
                        "id": 1
                    },
                    "feed": {
                        "id": 1,
                        "user": {
                            "id": 1
                        }
                    }
                }
            ],
            "all_comment": [
                {
                    "id": 1,
                    "user": {
                        "id": 2
                    },
                    "feed": {
                        "id": 1,
                        "user": {
                            "id": 1
                        }
                    }
                }
            ]
        }
    }
]

结果在 Jack 所有的 Feed 基础上,添加了一个 all_comment 字段,而这个字段正好是过滤了 Jack 的评论,是我要的最终结果。

原文地址:获取一些包含多对多外键但是不包含特定另一个外键的结果
我的博客:时空路由器

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 原文:https://my.oschina.net/liuyuantao/blog/751438 查询集API 参...
    阳光小镇少爷阅读 9,230评论 0 8
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,323评论 19 139
  • 一.自我介绍 一个人的自我介绍是给别人除视觉以外的另一个刺激。那怎么算是抓住别人眼球的自我介绍呢?我从今天64位伙...
    6玲子6阅读 2,338评论 2 3
  • 1、筱晓今天发烧好了很多,精神也好了不少。总结起来应该还是跟积食有很大的关系。第一次生病,没有什么经验,还好小伙伴...
    玉露君阅读 3,332评论 2 0
  • PS: 以色列哲学家马丁.布伯,二十世纪最重要的哲学家,宗教哲学。《我与你》、《人与人之间》、《两种类型的信仰》、...
    涂涂悦读阅读 3,496评论 0 0