Django ORM SELECT с объединением - PullRequest
       28

Django ORM SELECT с объединением

9 голосов
/ 26 октября 2011

ERR

Модели в Django:

class Key(models.Model):
    id     = models.AutoField(primary_key=True, blank=True)
    name   = models.CharField(max_length=50)


class Record(models.Model):
    id         = models.AutoField(primary_key=True, blank=True)
    project_id = models.IntegerField()
    name       = models.CharField(max_length=50)


class Value(models.Model):
    id        = models.AutoField(primary_key=True, blank=True)
    record    = models.ForeignKey(Record)
    key       = models.ForeignKey(Key)
    value     = models.CharField(max_length=255)

Мне нужно выбрать из БД эти данные:

NAME (from record)
and fields related with this record
[NAME (from key), VALUE (from value)]
[NAME (from key), VALUE (from value)]
[...]

Могу ли я использовать django ORM, чтобы сделать этот выбор? (например, в SQL select выглядит так)

SELECT
    `keeper_record`.`id` AS `record_id`,
    `keeper_record`.`name` AS `name`,
    `keeper_record`.`desc` AS `desc`,
    `keeper_key`.`name` AS `key_name`,
    `keeper_key`.`desc` AS `key_desc`,
    `keeper_value`.`value` AS `value_value`
FROM `keeper_record`
JOIN `keeper_value` ON `keeper_record`.`id` = `keeper_value`.`record_id`
JOIN `keeper_key` ON `keeper_key`.`id` = `keeper_value`.`key_id`
WHERE record_id = id

Ответы [ 2 ]

8 голосов
/ 26 октября 2011

Далее выбираются значения, связанные с конкретным идентификатором записи.Затем вы можете следовать внешним ключам, чтобы получить соответствующую запись и ключ.Использование select_related минимизирует поиск в базе данных.

# Select all values related to a record in your view
record = Record.objects.get(pk=record_id)
values = Value.objects.filter(record=record).select_related()

# In your template
{% for value in values %}
{{ value.record.name }} - {{ value.key.name }} - {{ value.value }}
{% endfor %}

Выбор более одной записи

В вашем sql у вас было WHERE record_id = 1, поэтому я показал, как получить все значения для определенной записи.Вы также можете выбрать значения для более чем одной записи в одном запросе.

# filter all records which belong to the project with `project_id=1`
records = Record.objects.filter(project_id=1)
# select all values that belong to these records
values = Value.objects.filter(record__in=records).select_related().order_by('record')
2 голосов
/ 26 октября 2011

Должно быть относительно простым, поскольку у вас уже есть внешние ключи, связанные вместе.

Record.objects.select_related().filter(id = variable_that_stores_id)

Вы можете комбинировать это только с , чтобы ограничить поля, которые вы хотите вернуть.

...