對於Yii2.0表關聯查詢的分析

來源:互聯網
上載者:User
這篇文章主要介紹了Yii2.0表關聯查詢的方法,結合執行個體形式分析了Yii中關聯查詢的實現方法與相關提示,需要的朋友可以參考下

本文執行個體講述了Yii2.0表關聯查詢的方法。分享給大家供大家參考,具體如下:

你可以使用 ActiveRecord 來進行關聯查詢(比如,從A表讀取資料時把關聯的B表資料也一起讀出來), 在Active Record中,擷取關聯資料可以像訪問主表ActiveRecord對象的屬性(property)一樣簡單。

比如,通過合適的關係聲明,你可以使用 $customer->orders 來擷取一個 Order 對象數組,代表該客戶下的訂單。

要聲明一個關係(relation),定義一個getter方法,該方法返回一個 yii\db\ActiveQuery 對象,擁有關聯上下文資訊,這樣將只查詢合格相關資料。比如:

class Customer extends \yii\db\ActiveRecord{ public function getOrders() {  // Customer has_many Order via Order.customer_id -> id  return $this->hasMany(Order::className(), ['customer_id' => 'id']); }}class Order extends \yii\db\ActiveRecord{ // Order has_one Customer via Customer.id -> customer_id public function getCustomer() {  return $this->hasOne(Customer::className(), ['id' => 'customer_id']); }}

上述代碼中的 yii\db\ActiveRecord::hasMany() 和 yii\db\ActiveRecord::hasOne() 是用來建模關係型資料庫中的 一對多 以及 一對一 關聯關係。比如,一個客戶customer有多個訂單orders,而一個訂單擁有或歸屬於一個使用者。兩個方法均接收兩個參數並返回一個 yii\db\ActiveQuery 對象:

$class: 關聯模型的類名稱。

$link: 兩張表之間的列關聯。這得是一個數組。數組元素的鍵是 $class 所對應表的列名稱,而數組元素的值是當前聲明類的列名稱。依據表外部索引鍵關聯來定義這些關係是一個好的編程實踐。

完成上述聲明後,就可以通過定義相應的getter方法來像訪問對象屬性一樣擷取關聯資料:

// get the orders of a customer$customer = Customer::findOne(1);$orders = $customer->orders; // $orders is an array of Order objects

上述代碼在幕後實際執行了如下兩個SQL查詢,分別對應於上述兩行代碼:

SELECT * FROM customer WHERE id=1;SELECT * FROM order WHERE customer_id=1;

提示:如果你再次訪問 $customer->orders ,並不會重複執行上述第2行SQL查詢。這條查詢語句只在運算式第一次被訪問時才被執行。後續的訪問將直接返回內部緩衝資料。如果你想重新執行查詢,只需要先調用一下unset來清除緩衝:

unset($customer->orders);.

有時候,你可能想傳遞參數給關聯查詢來限定查詢條件。比如只想讀取超過指定數額的大額訂單,而不是所有訂單。為此,可以使用如下getter方法來聲明一個 bigOrders 關係:

class Customer extends \yii\db\ActiveRecord{ public function getBigOrders($threshold = 100) {  return $this->hasMany(Order::className(), ['customer_id' => 'id'])   ->where('subtotal > :threshold', [':threshold' => $threshold])   ->orderBy('id'); }}

記住 hasMany() 返回對象是一個 yii\db\ActiveQuery ,因此ActiveQuery的方法都可以被用來定製這個關聯查詢。

通過上述聲明,如果你訪問 $customer->bigOrders, 它將只返回數額大於100的訂單。如果想指定一個不同的限定值,使用如下代碼:

$orders = $customer->getBigOrders(200)->all();

注意:關聯方法返回一個 yii\db\ActiveQuery 執行個體。如果你以屬性(類property)的方式來訪問它,返回資料是一個 yii\db\ActiveRecord 執行個體、或者是ActiveRecord數組或為空白(null)。比如, $customer->getOrders() 返回一個 ActiveQuery 執行個體,而$customer->orders 返回一個 Order 對象數組(或者是一個空數組,如果查詢結果為空白)。

中間表關聯查詢

有時候,一些資料表通過中間表(pivot table)關聯在一起。為了聲明這樣的關係,我們可以定製 yii\db\ActiveQuery 對象,通過調用它的 via() 或 viaTable() 方法。

比如,如果訂單表 order 和商品表 item 通過串連表 order_item關聯,我們可以在 Order 類中聲明 items 關係如下:

class Order extends \yii\db\ActiveRecord{ public function getItems() {  return $this->hasMany(Item::className(), ['id' => 'item_id'])   ->viaTable('order_item', ['order_id' => 'id']); }}

via() 方法和 viaTable() 類似,不過第一個參數是在當前ActiveRecord類中聲明的一個關係(relation)名,而不是中間表的名稱。比如,上述 items 關係也可以用下面的方法進行聲明:

class Order extends \yii\db\ActiveRecord{ public function getOrderItems() {  return $this->hasMany(OrderItem::className(), ['order_id' => 'id']); } public function getItems() {  return $this->hasMany(Item::className(), ['id' => 'item_id'])   ->via('orderItems'); }}

以上就是本文的全部內容,希望對大家的學習有所協助,更多相關內容請關注topic.alibabacloud.com!

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.