2013-02-15 75 views
7

如何在Django中预取相关对象并按中间表中的字段排序?Django:根据中间表字段排序的prefetch_related结果

这里是我的工作模型:

class Node(models.Model): 
    name = models.CharField(max_length=255) 
    edges = models.ManyToManyField('self', through='Edge', symmetrical=False) 


class Edge(models.Model): 
    from_node = models.ForeignKey(Node, related_name='from_node') 
    to_node = models.ForeignKey(Node, related_name='to_node') 

    weight = models.FloatField(default=0) 

给定一个节点,我想预取所有相关节点的重量排序。

当我使用此查询:

n = Node.objects.prefetch_related('to_node').order_by('edge__weight').get(name='x') 

的ORDER_BY没有效果。

编辑:

我最好的答案至今

n = Node.objects.get(name='x') 
edges = Edge.objects.filter(from_node=n).prefetch_related('to_node').order_by('weight') 

然后,而不是迭代n.edges(因为我喜欢),我重复edges.to_node

+0

您正在将节点表上的order by子句放在边缘表上。所以,所提到的结果是可以预料的。 – 2013-02-15 23:19:49

+0

确实,但它怎么能放在边桌上呢? – 2013-02-15 23:29:26

回答

4

只是一个概念性的想法(从记忆中写出)。

问题是,order_by引用了Node模型。

然而,有一种方法可以

Node.objects.get(name='x').edges.extra(select={'weight':'%s.weight' % Edge._meta.db_table}).order_by('weight') 

这将迫使ORM到:

  1. 添加 '重量' 领域,这通常会被省略。
  2. 按顺序排列结果。

查询次数应该与prefetch_query工作时相同,一次获取节点,第二次获取相关节点。

不幸的是,这不是一个非常'干净'的解决方案,因为我们需要使用_meta

+1

可能不干净,但我可以确认它确实有效。谢谢! – 2013-02-16 18:12:54

0

不就是干净虽然..

//Untested Code 
Node n = Node.objects.get(name="x") 

//This would return To Node IDs' ordered by weight 

n.edges.filter(from_node = n).values_list('to_node', flat=True).order_by('weight') 
5

如今,你还可以使用预取类来实现这一目标:

https://docs.djangoproject.com/en/1.10/ref/models/querysets/#django.db.models.Prefetch

或者,如果你想要做这一切的时候作为默认,你可以看看在元订购中间表,类似于:

class SomeThroughModel(models.Model): 
    order = models.IntegerField("Order", default=0, blank=False, null=False) 
    ... 

    class Meta: 
     ordering = ['order'] # order is the field holding the order 
相关问题