Хобрук: Ваш путь к мастерству в программировании

Как выбрать минимальное значение с помощью leftJoin в Doctrine с помощью построителя запросов

Попытка сделать leftJoin, получая самое низкое значение цены во втором с помощью groupBy в первой таблице:

        ->select('a', 'ap')
        ->from('AppBundle:Accomodation', 'a')
        ->leftJoin('a.rooms', 'ap')
        ->groupBy ('a.id')
        ->orderBy('a.name','ASC')
        ->getQuery()->getResult();

И когда я пытаюсь сделать это ниже, он не извлекает данные (ошибка trhows на Twig: поле a.name не существует):

        ->select('a', $qb->expr()->min('ap.price'))
        ->from('AppBundle:Accomodation', 'a')
        ->leftJoin('a.rooms', 'ap')
        ->groupBy ('a.id')
        ->orderBy('a.name','ASC')
        ->getQuery()->getResult();

Аннотации в обоих объектах таковы:

Размещение.php

/**
 * @ORM\OneToMany(targetEntity="AccomodationRoom", mappedBy="accomodation")
 */
private $rooms;

РазмещениеRoom.php

/**
 * @ORM\ManyToOne(targetEntity="Accomodation", inversedBy="rooms")
 * @ORM\JoinColumn(name="accomodation_id", referencedColumnName="id")
 */
private $accomodation;

Любые идеи? Спасибо!

25.09.2016

  • Предположительно свойство name существует в сущности AppBundle:Accomodation? 26.09.2016
  • Точно, существует... 26.09.2016
  • Пожалуйста, не могли бы вы показать нам часть схемы базы данных и объект, который относится к name 27.09.2016
  • Но сообщение об ошибке жалуется на a.name, не могли бы вы показать нам эту часть. 27.09.2016
  • имя - это просто строковое поле размещения, оно извлекает данные нормально с первой попытки, но не со второй 27.09.2016

Ответы:


1

Я думаю, что, возможно, в первом примере происходит то, что если ваша сущность Accomodation имеет свойство "комнаты", тогда доктрина не будет заполнять результаты, используя отдельные ключи массива для "а" и "ап", потому что "комнаты" возвращаются в родительской сущности "проживание".

$result = $query
    ->select('user', 'posts')
    ->from('MyBundle:User', 'user')
    ->leftJoin('a.posts', 'posts')
    ->groupBy ('user.id')
    ->orderBy('user.name','ASC')
    ->getQuery()->getResult();

В результате получится одномерный массив пользователей. Я даже не думаю, что вам нужно левое соединение, поскольку доктрина автоматически загрузит связанный объект, когда вы попытаетесь получить к нему доступ.

Теперь во втором примере (сломанном) второй аргумент для select() больше не является связанным объектом, а является скалярным значением. В этом случае Doctrine вернет многомерный массив.

$result = $query
    ->select('user', $qb->expr()->min('posts.length'))
    ->from('MyBundle:User', 'user')
    ->leftJoin('a.posts', 'posts')
    ->groupBy ('user.id')
    ->orderBy('user.name','ASC')
    ->getQuery()->getResult();

Имеет вид

[
    [
        0 => $user,
        1 => $minLength
    ],
    [
        0 => $user,
        1 => $minLength
    ],
    ...
]

Что сломает ваш шаблон ветки, так как вы зацикливаетесь на верхнем уровне массива. Чтобы сделать вещи лучше в twig, вы можете использовать псевдонимы в своем результате, например:

$query
    ->select('a as accomodation', 'MIN(ap.price) as minPrice')
    ->from('AppBundle:Accomodation', 'a')
    ->leftJoin('a.rooms', 'ap')
    ->groupBy ('a.id')
    ->orderBy('a.name','ASC')
    ->getQuery()->getResult();

дам тебе:

[
    [
        'accomodation' => $accomodation,
        'minPrice' => $minPrice
    ],
    [
        'accomodation' => $accomodation,
        'minPrice' => $minPrice
    ],
    ...
]

(Вы не можете использовать min как псевдоним, потому что это зарезервированное слово)

Затем вы можете изменить свой шаблон ветки, чтобы ссылаться на псевдонимы, что делает вещи более явными.

27.09.2016
  • Большой! это был двумерный массив. Самое простое решение в этом случае — это сделать так: {% set accomodation = item[0] %} прямо внутри цикла в Twig. Спасибо большое! Отличный и подробный ответ! 27.09.2016
  • Рад, что это помогло. Чтобы помочь вам с такими проблемами в будущем, я бы порекомендовал изучить отладку (xdebug или простой var_dump в Symfony и dump в twig). 27.09.2016
  • Новые материалы

    Создание кнопочного меню с использованием HTML, CSS и JavaScript
    Вы будете создавать кнопочное меню, которое имеет состояние наведения, а также позволяет вам выбирать кнопку при нажатии на нее. Финальный проект можно увидеть в этом Codepen . Шаг 1..

    Внедрите OAuth в свои веб-приложения для повышения безопасности
    OAuth — это широко распространенный стандарт авторизации, который позволяет приложениям получать доступ к ресурсам от имени пользователя, не раскрывая его пароль. Это позволяет пользователям..

    Классы в JavaScript
    class является образцом java Script Object. Конструкция «class» позволяет определять классы на основе прототипов с чистым, красивым синтаксисом. // define class Human class Human {..

    Как свинг-трейдеры могут использовать ИИ для больших выигрышей
    По мере того как все больше и больше профессиональных трейдеров и активных розничных трейдеров узнают о возможностях, которые предоставляет искусственный интеллект и машинное обучение для улучшения..

    Как построить любой стол
    Я разработчик программного обеспечения. Я люблю делать вещи и всегда любил. Для меня программирование всегда было способом создавать вещи, используя только компьютер и мое воображение...

    Обзор: Машинное обучение: классификация
    Только что закончил третий курс курса 4 часть специализации по машинному обучению . Как и второй курс, он был посвящен низкоуровневой работе алгоритмов машинного обучения. Что касается..

    Разработка расширений Qlik Sense с qExt
    Использование современных инструментов веб-разработки для разработки крутых расширений Вы когда-нибудь хотели кнопку для установки переменной в приложении Qlik Sense? Когда-нибудь просили..