Yii2.0表關(guān)聯(lián)查詢(xún)實(shí)例分析
本文實(shí)例講述了Yii2.0表關(guān)聯(lián)查詢(xún)的方法。分享給大家供大家參考,具體如下:
你可以使用 ActiveRecord 來(lái)進(jìn)行關(guān)聯(lián)查詢(xún)(比如,從A表讀取數(shù)據(jù)時(shí)把關(guān)聯(lián)的B表數(shù)據(jù)也一起讀出來(lái)), 在Active Record中,獲取關(guān)聯(lián)數(shù)據(jù)可以像訪問(wèn)主表ActiveRecord對(duì)象的屬性(property)一樣簡(jiǎn)單。
比如,通過(guò)合適的關(guān)系聲明,你可以使用 $customer->orders 來(lái)獲取一個(gè) Order 對(duì)象數(shù)組,代表該客戶(hù)下的訂單。
要聲明一個(gè)關(guān)系(relation),定義一個(gè)getter方法,該方法返回一個(gè) yii\db\ActiveQuery 對(duì)象,擁有關(guān)聯(lián)上下文信息,這樣將只查詢(xú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() 是用來(lái)建模關(guān)系型數(shù)據(jù)庫(kù)中的 一對(duì)多 以及 一對(duì)一 關(guān)聯(lián)關(guān)系。比如,一個(gè)客戶(hù)customer有多個(gè)訂單orders,而一個(gè)訂單擁有或歸屬于一個(gè)用戶(hù)。兩個(gè)方法均接收兩個(gè)參數(shù)并返回一個(gè) yii\db\ActiveQuery 對(duì)象:
$class: 關(guān)聯(lián)模型的類(lèi)名稱(chēng)。
$link: 兩張表之間的列關(guān)聯(lián)。這得是一個(gè)數(shù)組。數(shù)組元素的鍵是 $class 所對(duì)應(yīng)表的列名稱(chēng),而數(shù)組元素的值是當(dāng)前聲明類(lèi)的列名稱(chēng)。依據(jù)表外鍵關(guān)聯(lián)來(lái)定義這些關(guān)系是一個(gè)好的編程實(shí)踐。
完成上述聲明后,就可以通過(guò)定義相應(yīng)的getter方法來(lái)像訪問(wèn)對(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查詢(xún),分別對(duì)應(yīng)于上述兩行代碼:
SELECT * FROM customer WHERE id=1; SELECT * FROM order WHERE customer_id=1;
提示:如果你再次訪問(wèn) $customer->orders ,并不會(huì)重復(fù)執(zhí)行上述第2行SQL查詢(xún)。這條查詢(xún)語(yǔ)句只在表達(dá)式第一次被訪問(wèn)時(shí)才被執(zhí)行。后續(xù)的訪問(wèn)將直接返回內(nèi)部緩沖數(shù)據(jù)。如果你想重新執(zhí)行查詢(xún),只需要先調(diào)用一下unset來(lái)清除緩存:
unset($customer->orders);.
有時(shí)候,你可能想傳遞參數(shù)給關(guān)聯(lián)查詢(xún)來(lái)限定查詢(xún)條件。比如只想讀取超過(guò)指定數(shù)額的大額訂單,而不是所有訂單。為此,可以使用如下getter方法來(lái)聲明一個(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的方法都可以被用來(lái)定制這個(gè)關(guān)聯(lián)查詢(xún)。
通過(guò)上述聲明,如果你訪問(wèn) $customer->bigOrders, 它將只返回?cái)?shù)額大于100的訂單。如果想指定一個(gè)不同的限定值,使用如下代碼:
$orders = $customer->getBigOrders(200)->all();
注意:關(guān)聯(lián)方法返回一個(gè) yii\db\ActiveQuery 實(shí)例。如果你以屬性(類(lèi)property)的方式來(lái)訪問(wèn)它,返回?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ù)組,如果查詢(xún)結(jié)果為空)。
中間表關(guān)聯(lián)查詢(xún)
有時(shí)候,一些數(shù)據(jù)表通過(guò)中間表(pivot table)關(guān)聯(lián)在一起。為了聲明這樣的關(guān)系,我們可以定制 yii\db\ActiveQuery 對(duì)象,通過(guò)調(diào)用它的 via() 或 viaTable() 方法。
比如,如果訂單表 order 和商品表 item 通過(guò)連接表 order_item關(guān)聯(lián),我們可以在 Order 類(lèi)中聲明 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() 類(lèi)似,不過(guò)第一個(gè)參數(shù)是在當(dāng)前ActiveRecord類(lèi)中聲明的一個(gè)關(guān)系(relation)名,而不是中間表的名稱(chēng)。比如,上述 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)于Yii相關(guān)內(nèi)容感興趣的讀者可查看本站專(zhuān)題:《Yii框架入門(mén)及常用技巧總結(jié)》、《php優(yōu)秀開(kāi)發(fā)框架總結(jié)》、《smarty模板入門(mén)基礎(chǔ)教程》、《php面向?qū)ο蟪绦蛟O(shè)計(jì)入門(mén)教程》、《php字符串(string)用法總結(jié)》、《php+mysql數(shù)據(jù)庫(kù)操作入門(mén)教程》及《php常見(jiàn)數(shù)據(jù)庫(kù)操作技巧匯總》
希望本文所述對(duì)大家基于Yii框架的PHP程序設(shè)計(jì)有所幫助。
- Yii2增刪改查之查詢(xún) where參數(shù)詳細(xì)介紹
- Yii多表聯(lián)合查詢(xún)操作詳解
- YII2數(shù)據(jù)庫(kù)查詢(xún)實(shí)踐
- yii數(shù)據(jù)庫(kù)的查詢(xún)方法
- Yii2中多表關(guān)聯(lián)查詢(xún)hasOne hasMany的方法
- 詳解YII關(guān)聯(lián)查詢(xún)
- 詳解Yii2.0使用AR聯(lián)表查詢(xún)實(shí)例
- YII框架關(guān)聯(lián)查詢(xún)操作示例
- Yii框架連表查詢(xún)操作示例
- Yii框架數(shù)據(jù)庫(kù)查詢(xún)、增加、刪除操作示例
- Yii框架where查詢(xún)用法實(shí)例分析
相關(guān)文章
YII2框架中使用yii.js實(shí)現(xiàn)的post請(qǐng)求
本文給大家介紹的是簡(jiǎn)單分析下用yii2的yii\helpers\Html類(lèi)和yii.js實(shí)現(xiàn)的post請(qǐng)求的方法,非常的簡(jiǎn)單,有需要的小伙伴可以參考下2017-04-04
php微信公眾平臺(tái)開(kāi)發(fā)(一) 配置接口
這篇文章主要為大家詳細(xì)介紹了php微信公眾平臺(tái)開(kāi)發(fā)第一篇,微信公眾號(hào)配置接口,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-12-12
Zend Framework教程之Zend_Db_Table_Row用法實(shí)例分析
這篇文章主要介紹了Zend Framework教程之Zend_Db_Table_Row用法,詳細(xì)講述了Zend_Db_Table_Row的功能,并結(jié)合實(shí)例形式詳細(xì)分析了Zend_Db_Table_Row操作數(shù)據(jù)的相關(guān)技巧,需要的朋友可以參考下2016-03-03
ThinkPHP5實(shí)現(xiàn)JWT?Token認(rèn)證的過(guò)程(親測(cè)可用)
這篇文章主要介紹了ThinkPHP5實(shí)現(xiàn)JWT?Token認(rèn)證,首先composer先掛載阿里云鏡像,安裝JWT擴(kuò)展,本文給大家講解的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-10-10
生成隨機(jī)字符串和驗(yàn)證碼的類(lèi)的PHP實(shí)例
這篇文章主要介紹了生成隨機(jī)字符串和驗(yàn)證碼的類(lèi)的PHP實(shí)例,有需要的朋友可以參考一下2013-12-12
php curl模擬post請(qǐng)求小實(shí)例
使用php curl模擬post請(qǐng)求的小例子,提供大家學(xué)習(xí)一下2013-11-11
php簡(jiǎn)單的留言板與回復(fù)功能具體實(shí)現(xiàn)
留言板是在剛接觸php時(shí)用來(lái)學(xué)習(xí)的一個(gè)簡(jiǎn)單的應(yīng)用例子了,今天我再給初學(xué)php的朋友提供一個(gè)完整的php留言板的全部制作過(guò)程,希望對(duì)你會(huì)有幫助2014-02-02
Laravel創(chuàng)建數(shù)據(jù)庫(kù)表結(jié)構(gòu)的例子
今天小編就為大家分享一篇Laravel創(chuàng)建數(shù)據(jù)庫(kù)表結(jié)構(gòu)的例子,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-10-10
yii框架通過(guò)控制臺(tái)命令創(chuàng)建定時(shí)任務(wù)示例
這篇文章主要介紹了yii框架通過(guò)控制臺(tái)命令創(chuàng)建定時(shí)任務(wù)示例,需要的朋友可以參考下2014-04-04

