django 实现树形结构两种方式

树形结构类似于数据结构中的二叉树,如果树形结构比较复杂的话,通过递归的方式查询并不是那么简介,多次 访问数据库查询,对于性能是一个很大的损耗,最近看别人的算法,也就想要写一篇自己总结下,本文主要是在django上面如果集成这个算法,实现无限级分类。

basic classify

上面的例子中,我们列举了一个分类树,这个树一共有三级分类,从网上学到一个前序遍历树模型(The Nesat Set Model),该模型中通过每一层建立水平区间来区分不通的分类,一般上层的区间范围包含下级的区间。

原理如下:

水平区间分布

我们假定上衣分类的水平区间是1-20,那么风衣和T恤的分类在1-20之间分布,同理着装的分类区间应该大于上分区间的宽度(比如0-30)。

下面我们就为各个分类划定一个区间:

水平区间编号

从图中我们可以看到各个分类区间的范围:

类型 范围
着装 1-12
上衣 2-7
风衣 3-4
T恤 5-6
鞋子 8-11
运动鞋 9-10

添加一个新分类

假设我们需要添加一个裤子分类,那么裤子属于着装大分类下的二级分类,那么我们直接在区间的左边添加一类:

添加新区后的水平分布

接下来我们想办法来重新调整区间的范围,来为我们新的区间编号。

新区之后的编号变化

首先我们直接找到要添加的新区间的上级区间的最左边界m,然后找到所有的边界大于m的边界,进行统一的+2计算,接着以m+1, m+2作为边界插入新的区间。

删除一个分类

如果要删除一个分类,那么删除的可能是一个大的区间,区间后的所有区间都需要前移等量的一段:

删除一个区间

重排区间边界后:

删除区间后重排

从重排编号后,我们发现如果删除的区间对应的边界是m,n,那么所有大于n的边界都向前移动了n-m+1。

如何遍历一个分类

先来看看鞋子这个分类,鞋子本身是一个二级分类,鞋子下面还有一个三级分类运动鞋,那我们如何根据遍历鞋子这个分类。

  1. 先得到根结点的左右值(默认根节点的title为”顶级目录”)。
  2. 查询左右值在根节点的左右值范围内的记录,并且根据左值排序。
  3. 如果本次记录右值大于前次记录的右值则为平级分类

正常情况下,如果是一个只有两级的树,那么所有的第二季都是平级的,那么向栈中压入一个节点,在下一个节点的时候栈顶会被弹出,继续后面的节点扫描。

比如查询并排序的结果是:


[{'child': u'A', u'id': 14L, 'lft': 1L, 'parent': u'', 'rgt': 14L},
 {'child': u'D', u'id': 17L, 'lft': 2L, 'parent': u'A', 'rgt': 5L},
 {'child': u'H', u'id': 21L, 'lft': 3L, 'parent': u'D', 'rgt': 4L},
 {'child': u'B', u'id': 16L, 'lft': 6L, 'parent': u'A', 'rgt': 13L},
 {'child': u'F', u'id': 19L, 'lft': 7L, 'parent': u'B', 'rgt': 8L},
 {'child': u'E', u'id': 18L, 'lft': 9L, 'parent': u'B', 'rgt': 12L},
 {'child': u'G', u'id': 20L, 'lft': 10L, 'parent': u'E', 'rgt': 11L}]

这种情况下,堆栈的情况如下:

[[u'A', u'D'],
 [u'A', u'D', u'H'],
 [u'A', u'B'],
 [u'A', u'B', u'F'],
 [u'A', u'B', u'E'],
 [u'A', u'B', u'E', u'G']]

具体是如何弄的,详细的可以参考下面的代码。

Django实例


########################################### signals.py
# -*- coding: utf-8 -*-
from django.db.models import F
from django.db.models.signals import pre_save, post_delete
from django.dispatch import receiver

from core.models import ClosesClass

@receiver(pre_save, sender=ClosesClass)
def handle_save_closes_instance(sender, instance, **kwargs):
    """
    make other operation when insert new instance

    :param sender:
    :param instance:
    :param kwargs:
    :return:
    """
    if not instance.id:
        parent = sender.objects.filter(child=instance.parent)
        if not parent.exists():
            instance.lft = 1
            instance.rgt = 2
        else:
            lft = parent.first().lft
            sender.objects.filter(lft__gt=lft).update(lft=F("lft") + 2)
            sender.objects.filter(rgt__gt=lft).update(rgt=F("rgt") + 2)
            instance.lft = lft + 1
            instance.rgt = lft + 2

@receiver(post_delete, sender=ClosesClass)
def handler_delete_instance(sender, instance, **kwargs):
    """
    delete redundancy data

    :param sender:
    :param instance:
    :param kwargs:
    :return:
    """
    lft, rgt = instance.lft, instance.rgt
    width = rgt - lft + 1
    sender.objects.filter(lft__gt=rgt).update(lft=F("lft") - width)
    sender.objects.filter(rgt__gt=rgt).update(rgt=F("rgt") - width)

######################################## models.py
class ClosesClassManager(models.Manager):
    def get_queryset(self):
        return super(ClosesClassManager, self).get_queryset()

    def get_all_leaves_class(self):
        """
        获取所有的叶子节点

        :return:
        """
        return self.get_queryset() \
            .filter(rgt=F("lft") + 1) \
            .values_list("child", flat=True)

    def get_direct_children_of_class(self, parent):
        """
        获取直属的二级分类

        :param parent: parent class
        :return:
        """
        return self.get_queryset() \
            .filter(parent=parent) \
            .order_by("lft") \
            .values_list("child", flat=True)

    def get_all_class(self):
        """
        获取分类树
        :return:
        """

        # tree
        def convert_2_tree(paths):
            root = {}
            for path in paths:
                current_level = root
                for part in path:
                    if part not in current_level:
                        current_level[part] = {}
                    current_level = current_level[part]
            return root

        root_node = self.get_queryset().filter(lft=1).first()
        lft, rgt = root_node.lft, root_node.rgt
        closes = self.get_queryset().filter(lft__range=[lft, rgt]).order_by("lft").values()
        right = path_li = []
        for close in closes:
            if len(right):
                while right[-1]["rgt"] < close["rgt"]:
                    right = right[:-1]
            if len(right):
                title_li = [e["child"] for e in right]
                path_li.append(title_li + [close["child"]])
            right.append(close)
        return convert_2_tree(path_li)

class ClosesClass(models.Model):
    class Meta:
        db_table = "close_class"
        verbose_name_plural = u"衣服分类"

    # override bulk delete operation
    objects = ClosesClassManager()

    parent = models.CharField(max_length=128, blank=True, null=True, verbose_name=u"父级分类")
    child = models.CharField(max_length=128, blank=False, null=False, verbose_name=u"子级分类")
    lft = models.IntegerField(verbose_name=u"左边界")
    rgt = models.IntegerField(verbose_name=u"右边界")

# avoid circular import
from core.signals import *

第二种mptt

安装 django-mptt
pip install django-mptt

新建 area 应用
startapp area

models.py 里添加如下代码:

#coding:utf-8
from django.db import models
from mptt.models import MPTTModel

class Area(MPTTModel):
    name = models.CharField('名称', max_length=50, unique=True)
    parent = models.ForeignKey('self', verbose_name='上级区域', null=True, blank=True, related_name='children')

    class Meta:
        db_table = 'area'
        verbose_name = verbose_name_plural = '省/市/地区(县)'

    def __unicode__(self):
        return self.name

在这里只需让 Area 继承 MPTTModel
需要说明的是,实际上 MPTTModel 隐藏了四个变量:levellftrghttree_id。大多数时候我们是用不到这几个变量的。另外,如果你的 Modelparent 变量名字不是 "parent" 时,应当在 ModelMPTT 元类中指明:

#coding:utf-8
from django.db import models
from mptt.models import MPTTModel

class Area(MPTTModel):
    name = models.CharField('名称', max_length=50, unique=True)
    parent_area = models.ForeignKey('self', verbose_name='上级区域', null=True, blank=True, related_name='children')

    class Meta:
        db_table = 'area'
        verbose_name = verbose_name_plural = '省/市/地区(县)'

    class MPTTMeta:
        parent_attr = 'parent_area'

    def __unicode__(self):
        return self.name

admin.py 里添加如下代码:

from django.contrib import admin
from .models import Area

class AreaAdmin(admin.ModelAdmin):
    list_display = ('id', 'name', 'parent', 'level')

admin.site.register(Area, AreaAdmin)

在项目的 settings.pyINSTALLED_APPS 里添加 mpttarea,之后执行数据库操作。之后在 admin 后台中按照层级关系添加省/市/区(县)数据。

[图片上传失败...(image-fd628e-1521187090684)]

views.py 里添加如下代码:

from django.shortcuts import render_to_response
from .models import Area

def tree(request):
    nodes = Area.objects.all()
    return render_to_response('tree.html', {'nodes': nodes})

templates 文件夹下添加模版文件 tree.html

{% load staticfiles %}
{% load mptt_tags %}
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Django 树形结构</title>
    <link rel="stylesheet" href="{% static "css/css.css" %}"/>
    <script src="{% static "js/jquery-1.11.2.min.js" %}"></script>
    <script src="{% static "js/SimpleTree.js" %}"></script>
    <script type="text/javascript">
        $(function(){
            $(".framework_nav").SimpleTree({

            });
        });
    </script>
</head>
<body>
    <div class="framework_nav">
        <ul>
            <li class="all">
                <a href="#">省/市/地区(县)</a>
            </li>
            <ul show="true">
                {% recursetree nodes %}
                    <li>
                        {% if node.is_leaf_node %}
                            <li><a href="#">{{ node.name }}</a></li>
                        {% else %}
                            <li><a href="#">{{ node.name }}</a></li>
                            <ul>
                                <li><a href="#">{{ children }}</a></li>
                            </ul>
                        {% endif %}
                    </li>
                {% endrecursetree %}
            </ul>
        </ul>
    </div>
</body>
</html>

别忘了在 settings.py 里添加如下代码:

STATIC_ROOT = os.path.join(BASE_DIR, 'static')
STATICFILES_DIRS = [
    ('css', os.path.join(STATIC_ROOT, 'css')),
    ('js', os.path.join(STATIC_ROOT, 'js')),
    ('img', os.path.join(STATIC_ROOT, 'img')),
]

否则获取不到静态资源!
最后在项目的 urls.py 里添加如下代码:

from area.views import tree

urlpatterns = [
    ...
    url(r'^tree/', tree),
    ...
]

这样,就可以通过访问 http://127.0.0.1:8000/tree/ 看到树形结构了。

[图片上传失败...(image-13c3e5-1521187090683)]

转自:
https://www.jianshu.com/p/cd5986ccba6b

http://infinite.36deep.com/django-tree-storage

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,222评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,455评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,720评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,568评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,696评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,879评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,028评论 3 409
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,773评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,220评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,550评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,697评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,360评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,002评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,782评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,010评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,433评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,587评论 2 350

推荐阅读更多精彩内容