当前位置: 首页 > article >正文

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"}
        }
        """

http://www.kler.cn/news/285085.html

相关文章:

  • 鸿蒙HarmonyOS之使用preferences首选项保存获取数据
  • 1、Java简介+DOS命令+java的编译运行(字节码/机器码、JRE/JVM/JDK的区别)+一个简单的Java程序
  • Linux 数据结构 树知识
  • shell小白学习记录
  • 如何将线程绑定到特定的CPU核
  • HarmonyOS开发实战( Beta5版)减小应用包大小
  • 【2024】Datawhale X 李宏毅苹果书 AI夏令营 Task2
  • Linux(CentOS 7)
  • element的el-date-picker组件实现只显示年月日时分,不显示秒
  • 2024最新VMware17安装Windows10详细记录
  • SQL进阶技巧:如何查询最近一笔有效订单? | 近距离有效匹配问题
  • 微信小程序 === 组件样式
  • WHAT - 一个 IP 地址与地理信息的关联
  • JAVA中如何自定义注解
  • Docker compose 安装 ELK
  • 【电力电子】单相并网逆变器
  • 在Vue2中使用WebSocket
  • C语言基础(二十一)
  • CSS3换装达人原理
  • 【Datawhale AI夏令营】从零上手CV竞赛Task3
  • 惠中科技PV-Wiper全自动光伏清洁系统,根治污染难题
  • 2024最详细Maven配置教程
  • Java算法之归并排序(Merge Sort)
  • 【Godot4.3】MarkDown解析和生成类 - MDdoc
  • 仿华为车机功能之--修改Launcher3,实现横向滑动桌面空白处切换壁纸
  • 在Ubuntu 20.04上安装MySQL的方法
  • 神经网络搭建实战与Sequential的使用
  • 南京观海微电子----VCC、 VDD、VSS、VEE 电压符号解释
  • <Rust>egui学习之小部件(八):如何在窗口中添加滑动条slider部件?
  • Vue.js入门系列(十九):深入理解和应用组件自定义事件