Как это сделать в Laravel, подзапрос, где в

Как я могу сделать этот запрос в Laravel:

SELECT 
    `p`.`id`,
    `p`.`name`, 
    `p`.`img`, 
    `p`.`safe_name`, 
    `p`.`sku`, 
    `p`.`productstatusid` 
FROM `products` p 
WHERE `p`.`id` IN (
    SELECT 
        `product_id` 
    FROM `product_category`
    WHERE `category_id` IN ('223', '15')
)
AND `p`.`active`=1

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


person Marc Buurke    schedule 29.05.2013    source источник


Ответы (10)


Рассмотрим этот код:

Products::whereIn('id', function($query){
    $query->select('paper_type_id')
    ->from(with(new ProductCategory)->getTable())
    ->whereIn('category_id', ['223', '15'])
    ->where('active', 1);
})->get();
person lukaserat    schedule 12.11.2013
comment
Принял этот ответ, вопрос устарел, так как он касался Laravel 3, и приходящие ответы относятся к Laravel 4, ответ ниже будет работать и для 4. - person Marc Buurke; 29.07.2014
comment
@lukaserat рассматриваемый запрос применяет проверку AND _1 _._ 2_ = 1 к таблицам продуктов, тогда как ваш запрос применяет ее к таблице ProductCategory .... верно ?? или я что-то упускаю ..? - person hhsadiq; 07.07.2015
comment
@hhsadiq да, это относится к ProductCategory. - person lukaserat; 08.07.2015
comment
Хороший подход с названием таблицы. Для меня это плюс - person Alwin Kesler; 08.11.2016
comment
Хорошо работает для laravel 5.5 - person Oleg Shakhov; 08.10.2019

Ознакомьтесь с расширенной документацией по разделу where для Fluent. Вот пример того, чего вы пытаетесь достичь:

DB::table('users')
    ->whereIn('id', function($query)
    {
        $query->select(DB::raw(1))
              ->from('orders')
              ->whereRaw('orders.user_id = users.id');
    })
    ->get();

Это произведет:

select * from users where id in (
    select 1 from orders where orders.user_id = users.id
)
person drewjoh    schedule 29.05.2013
comment
Это близко, и я уже некоторое время ломаю голову над подобными запросами. Но where_in (laravel 3) требует 2 аргумента, второй из которых является массивом. Есть идеи, как это сделать правильно? Кроме того, я не думаю, что laravel 3 поддерживает метод from. - person Marc Buurke; 29.05.2013
comment
Ах, Laravel3 ... Ага, тогда это будет сложно. И я думаю, что в Laravel3 вы используете метод table() вместо from(). У меня не было такой ситуации в L3, извините! - person drewjoh; 29.05.2013
comment
Я не вижу метод whereIn, который принимает лямбду в Illuminate \ Database \ Query \ Builder, он был переименован в whereSub? - person nbransby; 12.11.2013

Вы можете использовать переменную, используя ключевое слово "use ($ category_id)"

$category_id = array('223','15');
Products::whereIn('id', function($query) use ($category_id){
   $query->select('paper_type_id')
     ->from(with(new ProductCategory)->getTable())
     ->whereIn('category_id', $category_id )
     ->where('active', 1);
})->get();
person Ramesh    schedule 29.07.2014

Вы можете использовать Eloquent в различных запросах и упростить понимание и обслуживание:

$productCategory = ProductCategory::whereIn('category_id', ['223', '15'])
                   ->select('product_id'); //don't need ->get() or ->first()

а затем складываем все вместе:

Products::whereIn('id', $productCategory)
          ->where('active', 1)
          ->select('id', 'name', 'img', 'safe_name', 'sku', 'productstatusid')
          ->get();//runs all queries at once

Это сгенерирует тот же запрос, который вы написали в своем вопросе.

person Philipe    schedule 13.11.2018

У меня сработал следующий код:

$result=DB::table('tablename')
->whereIn('columnName',function ($query) {
                $query->select('columnName2')->from('tableName2')
                ->Where('columnCondition','=','valueRequired');

            })
->get();
person Aditya Singh    schedule 19.01.2017

Скрипт протестирован в Laravel 5.x и 6.x. Замыкание static может улучшить производительность в некоторых случаях.

Product::select(['id', 'name', 'img', 'safe_name', 'sku', 'productstatusid'])
            ->whereIn('id', static function ($query) {
                $query->select(['product_id'])
                    ->from((new ProductCategory)->getTable())
                    ->whereIn('category_id', [15, 223]);
            })
            ->where('active', 1)
            ->get();

генерирует SQL

SELECT `id`, `name`, `img`, `safe_name`, `sku`, `productstatusid` FROM `products` 
WHERE `id` IN (SELECT `product_id` FROM `product_category` WHERE 
`category_id` IN (?, ?)) AND `active` = ?
person Madan Sapkota    schedule 24.01.2020
comment
Есть ли способ использовать область ProductCategory? - person Sergei Krivosheenko; 24.05.2021

Laravel 4.2 и выше может использовать запрос отношения попытки: -

Products::whereHas('product_category', function($query) {
$query->whereIn('category_id', ['223', '15']);
});

public function product_category() {
return $this->hasMany('product_category', 'product_id');
}
person LC Yoong    schedule 20.01.2017

Вот мой подход к Laravel 8.x, собранный из нескольких ответов здесь:

Product::select(['id', 'name', 'img', 'safe_name', 'sku', 'productstatusid'])
    ->whereIn('id', ProductCategory::select(['product_id'])
        ->whereIn('category_id', ['223', '15'])
    )
    ->where('active', 1)
    ->get();
person Darkproduct    schedule 08.04.2021

используя переменную

$array_IN=Dev_Table::where('id',1)->select('tabl2_id')->get();
$sel_table2=Dev_Table2::WhereIn('id',$array_IN)->get();
person a3rxander    schedule 21.04.2019
comment
Это приведет к появлению двух запросов, а не только одного. Вы можете вставить первый построитель запросов во второй, не получая результата первого запроса. См. Ответ Филипса для получения более подробной информации. - person Darkproduct; 08.04.2021

Попробуйте этот онлайн-инструмент sql2builder

DB::table('products')
    ->whereIn('products.id',function($query) {
                            DB::table('product_category')
                            ->whereIn('category_id',['223','15'])
                            ->select('product_id');
                        })
    ->where('products.active',1)
    ->select('products.id','products.name','products.img','products.safe_name','products.sku','products.productstatusid')
    ->get();
person liquid207    schedule 11.11.2019