TypeError: Obiekt „RelatedManager” nie jest iterowalny

85

Django

Mam kolejne modele:

class Group(models.Model):
    name = models.CharField(max_length=100)
    parent_group = models.ManyToManyField("self", blank=True)

    def __unicode__(self):
        return self.name


class Block(models.Model):

    name = models.CharField(max_length=100)
    app = models.CharField(max_length=100)
    group = models.ForeignKey(Group)

    def __unicode__(self):
        return self.name

powiedzmy, blok b1 ma grupę g1 . Według jego nazwy chcę pobrać wszystkie bloki z grupy g1 . Napisałem następną funkcję rekurencyjną:

def get_blocks(group):

    def get_needed_blocks(group):
        for block in group.block_set:
            blocks.append(block)

        if group.parent_group is not None:
            get_needed_blocks(group.parent_group)

    blocks = []
    get_needed_blocks(group)
    return blocks

ale b1.group.block_set zwraca mi obiekt RelatedManager , który nie jest iterowalny.

Co robić? Co jest nie tak?

megido
źródło

Odpowiedzi:

175

Spróbuj tego:

block in group.block_set.all()
Andrey Fedoseev
źródło
9
Czy ktoś wie, dlaczego RelatedManager nie jest stworzony do iterowalności? Spodziewałbym się, że tak będzie ...
stalepretzel
2
Ten komentarz jest trochę spóźniony, ale rozumowanie jest takie, że możesz użyć metod filtrowania w menedżerze, aby ograniczyć zapytanie przed wykonaniem pobierania do bazy danych.
owinąć
8
@stalepretzel, ponieważ działa dokładnie jak standardowy menedżer django. Nie wywołujesz Model.objects i nie iterujesz po nim. Wywołujesz Model.objects.all (), a następnie wykonujesz iterację. Menedżer Django = / = zestaw zapytań Django.
Saturnix
a jeśli używasz go w szablonie pętli for, możesz zrobić {% dla obiektu w objects.other_object_set.all%}
Bruce
@stalepretzel ... zwłaszcza, że ​​nazywa się to „block_SET” - a nie „blocks_relatedmanager”. Lub „blocks_set_relatedmanager”.
Klaws
27

Użyj go jakManager . Jeśli chcesz wszystkie obiekty, wywołaj all()metodę.

Ignacio Vazquez-Abrams
źródło