给关系排序依据

时间:2018-09-21 21:40:24

标签: django django-orm

给出以下Django模型:

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

class Beacon(models.Model):
    room = models.ForeignKey(Room)
    uuid = models.UUIDField(default=uuid.uuid4)
    major = models.PostiveIntegerField(max_value=65536)
    minor = models.PositiveIntegerField(max_value=65536)

Beacon模型是与房间的蓝牙信标关系。

我要选择与给定的uuid, major, minor组合匹配的所有房间。

问题是,我想通过离我最近的信标订购房间。因此,我需要能够为每个信标动态分配一个值,然后对其进行排序。

使用Django ORM可以吗?在Django 1.8中?

注意-我将事先知道信标的顺序,我将使用它们在查询字符串中传递的顺序。因此,传递的第一个信标(uuid,major,minor)应该与Room QuerySet返回的第一个房间匹配。

我正在设想这样的事情,尽管我知道这行不通:

beacon_order = [
    beacon1 = 1,
    beacon0 = 2,
    beacon3 = 3,
]

queryset = Room.objects.annotate(beacon_order=beacon_order).\
               order_by('beacon_order')

2 个答案:

答案 0 :(得分:0)

如果您已经知道信标的顺序,则无需在QuerySet本身内进行排序。取一个称为beacon_list的有序列表,该列表按顺序包含信标的主键,例如索引0处的项目是最近的信标的主键,索引1处的项目是第二个最接近的信标的PK,依此类推。然后使用列表推导:

ordered_rooms = [Room.objects.get(pk=x) for x in beacon_list]

您也不必使用PK,您可以使用任何可识别数据库中给定对象的东西,例如name字段。

答案 1 :(得分:0)

看起来像这样:

from django.db.models import Case, Q, When

beacons = request.query_params.getlist('beacon[]')
query = Q()
order = []

for pos, beacon in enumerate(beacons):
    uuid, major, minor = beacon.split(':')

    query |= Q(
        beacon__uuid=uuid,
        beacon__major=major,
        beacon__minor=minor,
        )

    order.append(When(
        beacon__uuid=uuid,
        beacon__major=major,
        beacon__minor=minor,
        then=pos,
        ))

rooms = Room.objects.filter(query).order_by(Case(*order))