NOT EXISTS
было бы хорошим решением, если бы это оставалось небольшим проектом (или если бы вы могли наложить больше ограничений на пользователей, чтобы сохранить небольшой набор данных, возможно, путем ограничения на основе даты создания пользователя или аналогичного).
Однако, если вы не можете этого сделать, этот запрос в конечном итоге вызовет проблемы, потому что под капотом mysql будет выполнять этот подзапрос для каждой записи, возвращаемой в основном запросе. Вы можете проверить https://dev.mysql.com/doc/refman/5.6/en/subquery-materialization.html для получения дополнительной информации.
Другое решение, которое немного лучше справилось бы с масштабированием, - это создание временной таблицы пользователей, у которых есть ваучер, который вы ищете для удаления ...
create temporary table tmp_voucher_user (user_id int not null, primary key (user_id)) as
select distinct user_id from voucher_users where voucher_id = 8;
Теперь, когда у нас есть таблица пользователей, которую нужно удалить, все, что нам нужно сделать, это позаботиться о простом левом соединении ...
select users.*, user_voucher.*
from users
inner join user_voucher on users.id = user_voucher.user_id
left join tmp_voucher_user on users.id = tmp_voucher_user.user_id
where tmp_voucher_user.user_id is null -- this part is important, it's only going to grab users where there isn't a match on tmp_user_voucher
К сожалению, это не так чисто, как просто выполнение NOT EXISTS
, и я не верю, что Laravel поддерживает способ создания временных таблиц помимо простого написания необработанного запроса, но он должен масштабироваться немного лучше.