DRF——serializer中获取嵌套评论

drf的serializer获取嵌套评论

获取嵌套评论的关键思路

在处理嵌套评论的序列化时,一个重要的思路是利用字典来存储和管理嵌套关系。在您提供的代码中,descendant_dict 用于存储当前根评论下所有子孙评论的序列化数据。这个字典的键是评论的ID,值是包含评论数据的字典。由于Python字典中的值是引用,而不是值的拷贝,因此当您在循环中向某个评论的 children 列表中添加数据时,实际上是在修改字典中存储的同一个列表对象。

这种内存地址的一致性意味着,当您在后续循环中再次访问或修改这个列表时,所有引用该列表的地方都会看到这些更改。这是一个高效的数据管理方式,因为它避免了不必要的数据复制,同时确保了数据的一致性和实时更新。

在您的代码中,通过这种方式,当您遍历所有子孙评论并构建它们的嵌套结构时,最终得到的 children_list 就已经包含了所有必要的嵌套层次。这种方法简化了数据的组织和遍历过程,使得序列化后的评论数据能够准确地反映出评论树的结构。

代码

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

    class Meta:
        model = models.Comment
        fields = ['create_datetime', 'reply', 'content', 'children']

    def get_children(self, obj):
        # print(obj, obj.content)  # obj表示正在处理的评论对象(循环queryset中的模型实例)
        # 序列化器会遍历 queryset 中的每一个模型实例。对于每一个实例,序列化器会调用 get_children 方法来获取其子孙评论

        # 获取以当前评论为根评论的子孙评论
        descendant_queryset = models.Comment.objects.filter(root=obj).order_by('id')

        descendant_dict = {}
        for descendant in descendant_queryset:
            ser = CreateCommentSerializer(instance=descendant, many=False)
            row = {'children': []}
            row.update(ser.data)
            descendant_dict[descendant.id] = row
        # print(descendant_dict)  # 包含当前根评论下所有子孙评论的字典
        '''
        {9: {'children': [], 'news': 3, 'content': '你说的挺好', 'depth': 1, 'reply': 3, 'create_datetime': '2024-08-31T10:00:27.583496'},
         10: {'children': [], 'news': 3, 'content': '或说八道', 'depth': 1, 'reply': 3, 'create_datetime': '2024-08-31T10:00:59.983649'}, 
         14: {'children': [], 'news': 3, 'content': '4防溺水抗菌素v防空识别空间', 'depth': 2, 'reply': 10, 'create_datetime': '2024-08-31T10:01:54.150888'}}
        '''

        # 存放当前根评论的1级评论
        children_list = []
        for cid, item in descendant_dict.items():
            depth = item['depth']
            if depth == 1:
                children_list.append(item)
                continue
            # 不是1级评论的,回复哪个放在哪个children中
            reply_id = item['reply']
            descendant_dict[reply_id]['children'].append(item)
            # 因为字典中存放的是每个item的地址(两者指向同一块内存空间),而不是直接拷贝一份
            # 所以当之后(下次循环)某个item的children新增数据时字典中的数据同样添加

        return children_list
        # 返回所有的1级评论(在经过以上对所有子孙评论循环,并放入各自回复评论的children中,1级评论就已经嵌套的包含所有评论了)
         """
        {
            11:{"reply": 2, children:[
                13->{,"reply": 11, children:[
                    15:{"reply": 13, children:[
                        16:{"reply": 15, children:[],"content": "oooadfa;skdjf;akjsd;flkjasdf","depth": 4, "create_datetime": "2021-09-01 22:32:22"}
                    ],"content": "oooadfa;skdjf;akjsd;flkjasdf","depth": 3, "create_datetime": "2021-09-01 22:32:22"}
                ],"content": "oooadfa;skdjf;akjsd;flkjasdf","depth": 2, "create_datetime": "2021-09-01 22:32:22"}
            ],"content": "oooadfa;skdjf;akjsd;flkjasdf","depth": 1, "create_datetime": "2021-09-01 22:32:22"}
            12:{"reply": 2, children:[
                14->{"reply": 12, children:[],"content": "oooadfa;skdjf;akjsd;flkjasdf","depth": 2, "create_datetime": "2021-09-01 22:32:22"}
            ],"content": "oooadfa;skdjf;akjsd;flkjasdf","depth": 1, "create_datetime": "2021-09-01 22:32:22"}


            13:{"reply": 11, children:[
                15:{"reply": 13, children:[],"content": "oooadfa;skdjf;akjsd;flkjasdf","depth": 3, "create_datetime": "2021-09-01 22:32:22"}
            ],"content": "oooadfa;skdjf;akjsd;flkjasdf","depth": 2, "create_datetime": "2021-09-01 22:32:22"}
            14:{"reply": 12, children:[],"content": "oooadfa;skdjf;akjsd;flkjasdf","depth": 2, "create_datetime": "2021-09-01 22:32:22"}
            15:{"reply": 13, children:[
                16:{"reply": 15, children:[],"content": "oooadfa;skdjf;akjsd;flkjasdf","depth": 4, "create_datetime": "2021-09-01 22:32:22"}
            ],"content": "oooadfa;skdjf;akjsd;flkjasdf","depth": 3, "create_datetime": "2021-09-01 22:32:22"}
            16:{"reply": 15, children:[],"content": "oooadfa;skdjf;akjsd;flkjasdf","depth": 4, "create_datetime": "2021-09-01 22:32:22"}
        }
        """
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

叫我DPT

你的鼓励将是我创作的最大动力

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

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

打赏作者

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

抵扣说明:

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

余额充值