溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Yii2.0中表關(guān)聯(lián)查詢的示例分析

發(fā)布時(shí)間:2021-08-30 14:56:35 來源:億速云 閱讀:123 作者:小新 欄目:開發(fā)技術(shù)

這篇文章將為大家詳細(xì)講解有關(guān)Yii2.0中表關(guān)聯(lián)查詢的示例分析,小編覺得挺實(shí)用的,因此分享給大家做個(gè)參考,希望大家閱讀完這篇文章后可以有所收獲。

具體如下:

你可以使用 ActiveRecord 來進(jìn)行關(guān)聯(lián)查詢(比如,從A表讀取數(shù)據(jù)時(shí)把關(guān)聯(lián)的B表數(shù)據(jù)也一起讀出來), 在Active Record中,獲取關(guān)聯(lián)數(shù)據(jù)可以像訪問主表ActiveRecord對(duì)象的屬性(property)一樣簡單。

比如,通過合適的關(guān)系聲明,你可以使用 $customer->orders 來獲取一個(gè) Order 對(duì)象數(shù)組,代表該客戶下的訂單。

要聲明一個(gè)關(guān)系(relation),定義一個(gè)getter方法,該方法返回一個(gè) yii\db\ActiveQuery 對(duì)象,擁有關(guān)聯(lián)上下文信息,這樣將只查詢符合條件的相關(guān)數(shù)據(jù)。比如:

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() 是用來建模關(guān)系型數(shù)據(jù)庫中的 一對(duì)多 以及 一對(duì)一 關(guān)聯(lián)關(guān)系。比如,一個(gè)客戶customer有多個(gè)訂單orders,而一個(gè)訂單擁有或歸屬于一個(gè)用戶。兩個(gè)方法均接收兩個(gè)參數(shù)并返回一個(gè) yii\db\ActiveQuery 對(duì)象:

$class: 關(guān)聯(lián)模型的類名稱。

$link: 兩張表之間的列關(guān)聯(lián)。這得是一個(gè)數(shù)組。數(shù)組元素的鍵是 $class 所對(duì)應(yīng)表的列名稱,而數(shù)組元素的值是當(dāng)前聲明類的列名稱。依據(jù)表外鍵關(guān)聯(lián)來定義這些關(guān)系是一個(gè)好的編程實(shí)踐。

完成上述聲明后,就可以通過定義相應(yīng)的getter方法來像訪問對(duì)象屬性一樣獲取關(guān)聯(lián)數(shù)據(jù):

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

上述代碼在幕后實(shí)際執(zhí)行了如下兩個(gè)SQL查詢,分別對(duì)應(yīng)于上述兩行代碼:

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

提示:如果你再次訪問 $customer->orders ,并不會(huì)重復(fù)執(zhí)行上述第2行SQL查詢。這條查詢語句只在表達(dá)式第一次被訪問時(shí)才被執(zhí)行。后續(xù)的訪問將直接返回內(nèi)部緩沖數(shù)據(jù)。如果你想重新執(zhí)行查詢,只需要先調(diào)用一下unset來清除緩存:

unset($customer->orders);.

有時(shí)候,你可能想傳遞參數(shù)給關(guān)聯(lián)查詢來限定查詢條件。比如只想讀取超過指定數(shù)額的大額訂單,而不是所有訂單。為此,可以使用如下getter方法來聲明一個(gè) bigOrders 關(guān)系:

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() 返回對(duì)象是一個(gè) yii\db\ActiveQuery ,因此ActiveQuery的方法都可以被用來定制這個(gè)關(guān)聯(lián)查詢。

通過上述聲明,如果你訪問 $customer->bigOrders, 它將只返回?cái)?shù)額大于100的訂單。如果想指定一個(gè)不同的限定值,使用如下代碼:

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

注意:關(guān)聯(lián)方法返回一個(gè) yii\db\ActiveQuery 實(shí)例。如果你以屬性(類property)的方式來訪問它,返回?cái)?shù)據(jù)是一個(gè) yii\db\ActiveRecord 實(shí)例、或者是ActiveRecord數(shù)組或?yàn)榭眨╪ull)。比如, $customer->getOrders() 返回一個(gè) ActiveQuery 實(shí)例,而$customer->orders 返回一個(gè) Order 對(duì)象數(shù)組(或者是一個(gè)空數(shù)組,如果查詢結(jié)果為空)。

中間表關(guān)聯(lián)查詢

有時(shí)候,一些數(shù)據(jù)表通過中間表(pivot table)關(guān)聯(lián)在一起。為了聲明這樣的關(guān)系,我們可以定制 yii\db\ActiveQuery 對(duì)象,通過調(diào)用它的 via() 或 viaTable() 方法。

比如,如果訂單表 order 和商品表 item 通過連接表 order_item關(guān)聯(lián),我們可以在 Order 類中聲明 items 關(guān)系如下:

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() 類似,不過第一個(gè)參數(shù)是在當(dāng)前ActiveRecord類中聲明的一個(gè)關(guān)系(relation)名,而不是中間表的名稱。比如,上述 items 關(guān)系也可以用下面的方法進(jìn)行聲明:

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

關(guān)于“Yii2.0中表關(guān)聯(lián)查詢的示例分析”這篇文章就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,使各位可以學(xué)到更多知識(shí),如果覺得文章不錯(cuò),請(qǐng)把它分享出去讓更多的人看到。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請(qǐng)聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

yii
AI