Получить все объекты, в которых есть ВСЕ выбранные элементы

У меня есть 3 таблицы:

Сертификаты - все доступные сертификаты

Пользователи - таблица пользователей

UserCertificates - сертификаты, которыми владеют пользователи. каждая запись имеет собственный ID, внешние ключи user_id и certificate_id.

Теперь у меня есть массив (или только одно значение) идентификаторов сертификатов, и мне нужно получить всех пользователей, владеющих этими выбранными сертификатами. Поэтому, если идентификаторы сертификатов 1,4,7, мне нужно выбрать всех пользователей, которым принадлежат ВСЕ эти сертификаты, а не 1 ИЛИ 4 ИЛИ 7.

У меня есть рабочий код, мне потребовалось несколько хороших часов, чтобы заставить его работать, но он не выглядит чистым. Есть ли более чистый способ сделать это?

$requiredCertificates - это массив идентификаторов сертификатов

public function getMatches(Opportunity $opportunity)
{
    $requiredCertificates = $this->getRequiredCertificates($opportunity);

    return User::with('certificates')
        ->whereHas('certificates', function ($query) use ($requiredCertificates) {
            $query->select(\DB::raw('count(distinct certificate_id)'))->whereIn('certificate_id', $requiredCertificates);
        }, '=', count($requiredCertificates))
        ->get();
}
Стоит ли изучать PHP в 2026-2027 годах?
Стоит ли изучать PHP в 2026-2027 годах?
Привет всем, сегодня я хочу высказать свои соображения по поводу вопроса, который я уже много раз получал в своем сообществе: "Стоит ли изучать PHP в...
Symfony Station Communiqué - 7 июля 2023 г
Symfony Station Communiqué - 7 июля 2023 г
Это коммюнике первоначально появилось на Symfony Station .
Оживление вашего приложения Laravel: Понимание режима обслуживания
Оживление вашего приложения Laravel: Понимание режима обслуживания
Здравствуйте, разработчики! В сегодняшней статье мы рассмотрим важный аспект управления приложениями, который часто упускается из виду в суете...
Установка и настройка Nginx и PHP на Ubuntu-сервере
Установка и настройка Nginx и PHP на Ubuntu-сервере
В этот раз я сделаю руководство по установке и настройке nginx и php на Ubuntu OS.
Коллекции в Laravel более простым способом
Коллекции в Laravel более простым способом
Привет, читатели, сегодня мы узнаем о коллекциях. В Laravel коллекции - это способ манипулировать массивами и играть с массивами данных. Благодаря...
Как установить PHP на Mac
Как установить PHP на Mac
PHP - это популярный язык программирования, который используется для разработки веб-приложений. Если вы используете Mac и хотите разрабатывать...
0
0
85
1

Ответы 1

Попробуй это..

public function getMatches(Opportunity $opportunity)
{
    $requiredCertificates = $this->getRequiredCertificates($opportunity);

    $users = User::with('certificates');

    foreach($requiredCertificates as $certificateId) {
        $users->whereHas('certificates', function ($query) use ($certificateId) {
            $query->where('certificate_id', $certificateId);
        });
    }

    return $users->get();
}

Сначала попробуйте выполнить return $ users-> toSql (), чтобы увидеть, что эта штука на самом деле создает. Хотя он выглядит чище (я тоже хотел это предложить, он был быстрее), он, возможно, более обременительный для базы данных. Должен увидеть sql

Inuyaki 15.03.2018 12:12

sql, созданный с использованием рассматриваемого метода (используется arry [1,2]): select * from 'users' where (select count(distinct certificate_id) from 'certificates' inner join 'user_certificates' on 'certificates'.'id' = 'user_certificates'.'certificates_id' where 'users'.'id' = 'user_certificates'.'user_id' and 'certificate_id' in (1, 2)) = 2

iamab.in 15.03.2018 12:55

sql, созданный с использованием рассматриваемого метода (используется arry [1,2]): select * from 'users' where exists (select * from 'certificates' inner join 'user_certificates' on 'certificates'.'id' = 'user_certificates'.'certificate_id' where 'users'.'id' = 'user_certificates'.'user_id' and 'certificate_id' = 1) and exists (select * from 'certificates' inner join 'user_certificates' on 'certificates'.'id' = 'user_certificates'.'certificate_id' where 'users'.'id' = 'user_certificates'.'user_id' and 'certificate_id' = 2)

iamab.in 15.03.2018 13:03

@Inuyaki, существует работает быстрее, чем in. Не так ли? (Я просто погуглил, но не уверен)

iamab.in 15.03.2018 17:25
связанный поиск
iamab.in 15.03.2018 18:23

Ваш метод примерно на 50% быстрее! Я использую github.com/barryvdh/laravel-debugbar с Laravel, и он добавляет в ваш проект панель инструментов, которая показывает все запросы, запущенные на странице, и время, необходимое для их обработки. Спасибо чувак! Теперь у меня есть другая проблема, поскольку я пытаюсь использовать это с плагином Datatables, и в настоящее время он выполняет 2 запроса на строку в таблице, поэтому, если у вас есть 50 строк, он выполнит 100 запросов ...

Varin 15.03.2018 19:31

существует быстрее, да ... Меня все еще интересует масштабируемость. Только для 2-х это явно быстрее, но что, если массив имеет длину 50 или около того? Почему у вас 2 запроса на строку в таблице?

Inuyaki 16.03.2018 07:51

Другие вопросы по теме