วิธีทำใน Laravel แบบสอบถามย่อยที่อยู่ใน


121

ฉันจะสร้างแบบสอบถามนี้ใน 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

ฉันสามารถทำได้ด้วยการเข้าร่วม แต่ฉันต้องการรูปแบบนี้เพื่อประสิทธิภาพ

คำตอบ:


199

พิจารณารหัสนี้:

Products::whereIn('id', function($query){
    $query->select('paper_type_id')
    ->from(with(new ProductCategory)->getTable())
    ->whereIn('category_id', ['223', '15'])
    ->where('active', 1);
})->get();

1
ยอมรับคำตอบนี้คำถามไม่ทันสมัยเนื่องจากเกี่ยวข้องกับ Laravel 3 และคำตอบที่เข้ามาเป็นของ Laravel 4 คำตอบด้านล่างจะใช้ได้กับ 4 เช่นกัน
Marc Buurke

3
@lukaserat pแบบสอบถามในคำถามที่ใช้และ active= 1 ตรวจสอบตารางผลิตภัณฑ์ในขณะที่แบบสอบถามของคุณใช้กับตาราง ProductCategory .... ใช่ไหม ?? หรือมีอะไรหาย .. ?
hhsadiq

@hhsadiq ใช่หมายถึง ProductCategory
lukaserat

1
แนวทางที่ดีกับชื่อตาราง เป็นข้อดีสำหรับฉัน
Alwin Kesler

ทำงานได้ดีสำหรับ laravel 5.5
Oleg Shakhov

53

ดูเอกสารประกอบขั้นสูงสำหรับ Fluent: http://laravel.com/docs/queries#advanced-wheres

นี่คือตัวอย่างของสิ่งที่คุณพยายามจะบรรลุ:

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
Marc Buurke

อา Laravel3 ... ใช่ว่าจะยากแล้ว และฉันคิดว่าใน Laravel3 คุณใช้table()วิธีนี้แทนไฟล์from(). ฉันไม่เคยมีสถานการณ์แบบนั้นใน L3 ขอโทษด้วย!
drewjoh

ฉันไม่เห็นเมธอด whereIn ซึ่งใช้ lambda ใน Illuminate \ Database \ Query \ Builder ได้รับการเปลี่ยนชื่อ whereSub หรือไม่
nbransby

20

คุณสามารถใช้ตัวแปรโดยใช้คำหลัก "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();

5

รหัสต่อไปนี้ใช้ได้ผลสำหรับฉัน:

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

            })
->get();

3

คุณสามารถใช้ 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

สิ่งนี้จะสร้างข้อความค้นหาเดียวกับที่คุณเขียนในคำถามของคุณ


2

สคริปต์ได้รับการทดสอบใน 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` = ?

1

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');
}

0
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();


-2

โปรดลองใช้เครื่องมือออนไลน์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();
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.