【问题标题】:How to do this in Laravel, subquery where in如何在 Laravel 中执行此操作,子查询在哪里
【发布时间】:2013-05-24 19:28:38
【问题描述】:

如何在 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

我也可以通过连接来做到这一点,但我需要这种格式来提高性能。

【问题讨论】:

    标签: php mysql laravel eloquent subquery


    【解决方案1】:

    查看 advanced where clause documentation 了解 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
    )
    

    【讨论】:

    • 接近了,我一直在为类似的查询困惑一段时间。但是 where_in (laravel 3) 需要 2 个参数,第二个是一个数组。知道如何做到这一点吗?另外,我认为 laravel 3 不支持 from 方法。
    • 啊,Laravel3... 是的,那会很困难。我认为在 Laravel3 中你使用table() 方法而不是from()。我在L3还没有遇到过这种情况,抱歉!
    • 我看不到在 Illuminate\Database\Query\Builder 中采用 lambda 的 whereIn 方法是否已重命名为 whereSub?
    【解决方案2】:

    这是我从这里的多个答案中收集的 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();
    

    【讨论】:

      【解决方案3】:

      脚本在 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` = ?
      

      【讨论】:

      • 有没有办法使用 ProductCategory 的范围?
      【解决方案4】:

      请试试这个在线工具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();
      

      【讨论】:

        【解决方案5】:

        使用变量

        $array_IN=Dev_Table::where('id',1)->select('tabl2_id')->get();
        $sel_table2=Dev_Table2::WhereIn('id',$array_IN)->get();
        

        【讨论】:

        • 这将导致两个查询,而不仅仅是一个。您可以将第一个查询构建器插入到第二个查询构建器中,而无需获得第一个 quary 的结果。有关详细信息,请参阅飞利浦的回答。
        【解决方案6】:

        您可以在不同的查询中使用 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
        

        这将生成您在问题中编写的相同查询。

        【讨论】:

          【解决方案7】:
          Product::from('products as p')
          ->join('product_category as pc','p.id','=','pc.product_id')
          ->select('p.*')
          ->where('p.active',1)
          ->whereIn('pc.category_id', ['223', '15'])
          ->get();
          

          【讨论】:

            【解决方案8】:

            Laravel 4.2 及更高版本,可以使用 try 关系查询:-

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

            【讨论】:

              【解决方案9】:

              以下代码对我有用:

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

              【讨论】:

                【解决方案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();
                

                【讨论】:

                • 接受了这个答案,问题已经过时了,因为它是关于 Laravel 3 的,答案是针对 Laravel 4,下面的答案也适用于 4。
                • @lukaserat 有问题的查询正在对产品表应用 AND p.active=1 检查,而您的查询将其应用于 ProductCategory 表......对吗?还是有什么我错过了..?
                • @hhsadiq 是的,它指的是 ProductCategory。
                • 使用表名的好方法。这对我来说是一个加分
                • 适用于 laravel 5.5
                【解决方案11】:

                你可以通过关键字“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();
                

                【讨论】:

                  猜你喜欢
                  • 2017-09-27
                  • 1970-01-01
                  • 1970-01-01
                  • 1970-01-01
                  • 1970-01-01
                  • 1970-01-01
                  • 2018-02-02
                  • 2020-05-30
                  • 1970-01-01
                  相关资源
                  最近更新 更多