DRF——serializer中获取嵌套评论

2024-08-31 23:36

本文主要是介绍DRF——serializer中获取嵌套评论,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

drf的serializer获取嵌套评论

获取嵌套评论的关键思路

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

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

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

代码

class ListCommentSerializer(serializers.ModelSerializer):children = serializers.SerializerMethodField()class Meta:model = models.Commentfields = ['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"}}"""

这篇关于DRF——serializer中获取嵌套评论的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1125312

相关文章

hdu1254(嵌套bfs,两次bfs)

/*第一次做这种题感觉很有压力,思路还是有点混乱,总是wa,改了好多次才ac的思路:把箱子的移动当做第一层bfs,队列节点要用到当前箱子坐标(x,y),走的次数step,当前人的weizhi(man_x,man_y),要判断人能否将箱子推到某点时要嵌套第二层bfs(人的移动);代码如下:

Android Environment 获取的路径问题

1. 以获取 /System 路径为例 /*** Return root of the "system" partition holding the core Android OS.* Always present and mounted read-only.*/public static @NonNull File getRootDirectory() {return DIR_ANDR

JS和jQuery获取节点的兄弟,父级,子级元素

原文转自http://blog.csdn.net/duanshuyong/article/details/7562423 先说一下JS的获取方法,其要比JQUERY的方法麻烦很多,后面以JQUERY的方法作对比。 JS的方法会比JQUERY麻烦很多,主要则是因为FF浏览器,FF浏览器会把你的换行也当最DOM元素。 <div id="test"><div></div><div></div

vcpkg子包路径批量获取

获取vcpkg 子包的路径,并拼接为set(CMAKE_PREFIX_PATH “拼接路径” ) import osdef find_directories_with_subdirs(root_dir):# 构建根目录下的 "packages" 文件夹路径root_packages_dir = os.path.join(root_dir, "packages")# 如果 "packages"

Weex入门教程之4,获取当前全局环境变量和配置信息(屏幕高度、宽度等)

$getConfig() 获取当前全局环境变量和配置信息。 Returns: config (object): 配置对象;bundleUrl (string): bundle 的 url;debug (boolean): 是否是调试模式;env (object): 环境对象; weexVersion (string): Weex sdk 版本;appName (string): 应用名字;

MFC中App,Doc,MainFrame,View各指针的互相获取

纸上得来终觉浅,为了熟悉获取方法,我建了个SDI。 首先说明这四个类的执行顺序是App->Doc->Main->View 另外添加CDialog类获得各个指针的方法。 多文档的获取有点小区别,有时间也总结一下。 //  App void CSDIApp::OnApp() {      //  App      //  Doc     CDocument *pD

android两种日志获取log4j

android   log4j 加载日志使用方法; 先上图: 有两种方式: 1:直接使用架包 加载(两个都要使用); 架包:android-logging-log4j-1.0.3.jar 、log4j-1.2.15.jar  (说明:也可以使用架包:log4j-1.2.17.jar)  2:对架包输入日志的二次封装使用; 1:直接使用 log4j 日志框架获取日志信息: A:配置 日志 文

17 通过ref代替DOM用来获取元素和组件的引用

重点 ref :官网给出的解释是: ref: 用于注册对元素或子组件的引用。引用将在父组件的$refs 对象下注册。如果在普通DOM元素上使用,则引用将是该元素;如果在子组件上使用,则引用将是组件实例: <!-- vm.$refs.p will be the DOM node --><p ref="p">hello</p><!-- vm.$refs.child will be the c

react笔记 8-19 事件对象、获取dom元素、双向绑定

1、事件对象event 通过事件的event对象获取它的dom元素 run=(event)=>{event.target.style="background:yellowgreen" //event的父级为他本身event.target.getAttribute("aid") //这样便获取到了它的自定义属性aid}render() {return (<div><h2>{

react笔记 8-18 事件 方法 定义方法 获取/改变数据 传值

1、定义方法并绑定 class News extends React.Component {constructor(props) {super(props)this.state = {msg:'home组件'}}run(){alert("我是一个run") //方法写在类中}render() {return (<div><h2>{this.state.msg}</h2><button onCli