首頁 後端開發 php教程 關於YII關聯查詢的解析

關於YII關聯查詢的解析

Jun 15, 2018 pm 01:51 PM
yii 關聯查詢

這篇文章主要介紹了YII關聯查詢的相關資料,需要的朋友可以參考下

一、多表關聯的配置

在我們使用AR 執行關聯查詢之前,我們需要讓AR 知道一個AR 類別是怎麼關聯到另一個的。

兩個 AR 類別之間的關係直接透過 AR 類別所代表的資料表之間的關係相關聯。從資料庫的角度來說,表A 和B 之間有三種關係:一對多(one-to-many,例如tbl_user 和tbl_post),一對一( one-to-one 例如tbl_user 和tbl_profile)和多對多(many-to-many 例如tbl_category 和tbl_post)。在AR 中,有四種關係:

BELONGS_TO(屬於): 如果表A 和B 之間的關係是一對多,則表B 屬於表A (例如Post 屬於User);

HAS_MANY(有多個): 如果表A 和B 之間的關係是一對多,則A 有多個B (例如User 有多個Post);

HAS_ONE(有一個) : 這是HAS_MANY 的一個特例,A 最多有一個B (例如User 最多有一個Profile);

MANY_MANY: 這個對應於資料庫中的多對多關係。由於多數 DBMS 不直接支援 多對多 關係,因此需要有一個關聯表將 多對多 關係分割為 一對多 關係。在我們的範例資料結構中,tbl_post_category 就是用於此目的的。在 AR 術語中,我們可以解釋MANY_MANY 為 BELONGS_TO 和 HAS_MANY 的組合。例如,Post 屬於多個(belongs to many) Category ,Category 有多個(has many) Post.

AR 中定義關係需要覆蓋 CActiveRecord 中的 relations() 方法。此方法傳回一個關係配置陣列。每個數組元素透過如下格式表示單一的關係。

'VarName'=>array('RelationType', 'ClassName', 'ForeignKey', ...additional options)
登入後複製

其中VarName 是關係的名字;RelationType 指定關係類型,可以是一下四個常數之一: self::BELONGS_TO, self::HAS_ONE, self::HAS_MANY and self::MANY_MANY;ClassName 是此AR 類別所關聯的AR 類別的名字; ForeignKey 指定關係中使用的外鍵(一個或多個)。

需要弄清楚的幾點:

(1),VarName指什麼?詳見下面例2。

(2),RelationType。共有4種,分別為

self::HAS_MANY, self::BELONGS_TO, self::MANY_MANY, self::HAS_ONE。

(3),ClassName。即關聯的另一個../model/類別名稱.php。

(4),ForeignKey。誰是誰的外鍵?

(5),附加條件

ER Diagram

#例1,一對多與多對一關係(post和user之間的關係)

1)models/Post.php

class Post extends CActiveRecord 
{ 
...... 
public function relations() 
{ 
return array( 
'author'=>array(self::BELONGS_TO, 'User', 'author_id'), 
); 
} 
}
登入後複製

其中Post與User的關係是BELONGS_TO(多對一)關係,並透過Post的author_id與User關聯。

Post中author_id是外鍵,關聯到User中。

註:此處的VarName是author,一個物件。

(2)models/User.php

class User extends CActiveRecord 
{ 
...... 
public function relations() 
{ 
return array( 
'posts'=>array(self::HAS_MANY, 'Post', 'author_id'), 
'profile'=>array(self::HAS_ONE, 'Profile', 'owner_id'), 
); 
} 
}
登入後複製

對於User,與Post的關係是屬於HAS_MANY(一對多)關係。並透過Post的author_id與Post關聯。

範例2,多對多重關係

在FailParts.php中

'Users' => array(self::MANY_MANY, 'User', 'fail_parts_user(fail_parts_id, user_id)'),
登入後複製

在User.php中

'FailParts' => array(self::MANY_MANY, 'FailParts', 'fail_parts_user(user_id, fail_parts_id)'),
登入後複製

#因為兩者是多對多關係,所以要用Users,而不是User;要用FailParts ,而不是FailPart。

此處的Users和FailParts,即為前面的VarName。

範例3,一對一關係

比較簡​​單,暫略。

2,關於VarName。

對於類別A.php,'VarName'=>array('RelationType', 'B', 'ForeignKey', ...additional options)
其中VarName與B基本相同。但未必完全一樣。此時就可以在A的views/A/xx.php中透過VarName來存取B及其屬性值了。

如果是一對一:A->VarName
如果是多對一:author_name = $post->Author->name;
如果一對多:$posts = $author->Post;
如果是多對多:$posts = $author->Post;//本質是拆成一對多對一

foreach($posts as $u){ 
$_tmp_titles[] = $u -> title; 
} 
titleStr = implode(', ', $_tmp_titles);
登入後複製

二、多表關聯的使用

#常常在controllers裡

1,延遲載入

(1)多對一

$post = Post::model()->findByPk(10); 
$author = $post->author;
登入後複製

批註:此處本質是一對一。

(2)一對多

$user = User::model()->findByPk(10); 
$posts = $user->posts;
登入後複製

(3)多對多

#需要重點注意:兩個id有先後關係。

站在$repairInfo實例的角度,關聯關係必須是

'FailParts' => array(self::MANY_MANY, 'FailParts', 'repair_mapping(repair_info_id,fail_parts_id)'),
登入後複製

而站在$failParts實例的角度,則關聯關係變為

'RepairInfos' => array(self::MANY_MANY, 'RepairInfo', 'repair_mapping(fail_parts_id, repair_info_id)'),
登入後複製

而前面也已經指出,不需要雙方都配置,只需需要的一方設定即可。

之前曾使用過的笨方法:

/*方法一:使用表关系(多对多)*/ 
$fails = $repairInfo->FailParts;//在$repairInfo中使用 
/*方法二:使用原始方法*/ 
$id = $repairInfo->id; 
$maps = RepairMapping::model()->findAll("repair_info_id = $id"); 
$f_ids = array(); 
foreach($maps as $map){ 
array_push($f_ids, $maps[0]->fail_parts_id); 
} 
$f_idsStr = implode(',',$f_ids); 
$fails = FailParts::model()->findAll("id IN ($f_idsStr)");
登入後複製

2,主動載入-with

#(1)一對多
(2)多對多

#
$posts = Post::model()->('author')->findAll();
登入後複製

例子:

User.php

//查询一个机房$idc_id的所有用户 
function getAdminedUsersByIdc($idc_id){ 
$c = new CDbCriteria(); 
$c->join = "JOIN idc_user on t.id=idc_user.user_id"; 
$c->condition = "idc_user.idc_id=$idc_id"; 
return User::model()->with('Idcs')->findAll($c); 
} 
//规则中配置 
'Idcs' => array(self::MANY_MANY, 'Idc', 'idc_user(user_id, idc_id)'),
登入後複製

批注:没有with('Idcs'),执行后的结果也一样。只不过不再是eager loading。

三、带参数的关联配置

常见的条件有

1,condition 按某个表的某个字段加过滤条件

例如:

//在User的model里定义,如下关联关系 
'doingOutsources' => array(self::MANY_MANY, 'Outsource', 'outsource_user(user_id, outsource_id)', 
'condition' => "doingOutsources.status_id IN(" . Status::ASSIGNED . "," . Status::STARTED ."," . Status::REJECTED .")"),
登入後複製

//结论:condition是array里指定model的一个字段。

显然,doingOutsources是真实数据表Outsource的别名,所以在condition中可以使用doingOutsources.status_id,当然也可以使用Outsource.status_id。另本表名user的默认别名是t。

2,order 按某个表的某个字段升序或降序

//在RepairInfo的model里定义,如下关联关系 
'WorkSheet' => array(self::HAS_MANY, 'WorkSheet', 'repair_info_id', order => 'created_at desc'), 
//调用 
$worksheets = $repair_info->WorkSheet; //此时$worksheets是按降序排列
登入後複製

//结论:order是array里指定model的一个字段。

with
joinType
select
params
on
alias
together
group
having
index

还有用于lazy loading的
limit 只取5个或10个
offset
through
官方手册
'posts'=>array(self::HAS_MANY, 'post', 'author_id', 'order'=>'posts.create_time DESC', 'with'=>'categories'),

四、静态查询(仅用于HAS_MANY和MANY_MANY)

关键字:self:STAT

1,基本用法。例如,

class Post extends CActiveRecord 
{ 
...... 

public function relations() 
{ 
return array( 
'commentCount'=>array(self::STAT, 'Comment', 'post_id'), 
'categoryCount'=>array(self::STAT,'Category','post_category(post_id, category_id)'); 

); 
} 
}
登入後複製

2,静态查询也支持上面的各种条件查询

'doingOutsourceCount' => array(self::STAT, 'Outsource', 'outsource_user(user_id, outsource_id)', 
'condition' => "outsource.status_id IN(" . Status::ASSIGNED . "," . Status::STARTED ."," . Status::REJECTED .")"),
登入後複製

其他查询还包括

condition 使用较多

order
select
defaultValue
params
group
having

3,静态查询的加载方式

可以使用lazy loading方式
$post->commentCount.
也可以使用eager loading方式
$posts = Post::model()->with('commentCount','categoryCount')->findAll();
注with中字符串一定是别名。

两者的性能比较:

如果需要取所有post的所有comment,前者需要2N+1次查询,而后者只有一次。两者的选择视情况而定。

以上就是本文的全部内容,希望对大家的学习有所帮助,更多相关内容请关注PHP中文网!

相关推荐:

Yii中表单用法实例

Yii2如何实现同时搜索多个字段

如何实现Yii2框架中使用PHPExcel导出Excel文件

以上是關於YII關聯查詢的解析的詳細內容。更多資訊請關注PHP中文網其他相關文章!

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

MyBatis一對多查詢配置詳解:解決常見關聯查詢問題 MyBatis一對多查詢配置詳解:解決常見關聯查詢問題 Feb 22, 2024 pm 02:18 PM

MyBatis一對多查詢配置詳解:解決常見關聯查詢問題,需要具體程式碼範例在實際的開發工作中,經常會遇到需要查詢主實體物件及其關聯的多個從實體物件的情況。在MyBatis中,一對多查詢是一種常見的資料庫關聯查詢,透過正確的配置,可以輕鬆實現對關聯物件的查詢、展示和操作。本文將介紹MyBatis中一對多查詢的設定方法,以及如何解決一些常見的關聯查詢問題,同時會

PHP中的多表關聯查詢技巧 PHP中的多表關聯查詢技巧 May 24, 2023 am 10:01 AM

PHP中的多表關聯查詢技巧關聯查詢是資料庫查詢的重要部分,特別是當你需要展示多個相關資料庫表內的資料時。在PHP應用程式中,在使用MySQL等資料庫時,多表關聯查詢經常會用到。多表關聯的含義是,將一個表中的資料與另一個或多個表中的資料進行比較,在結果中將那些滿足要求的行連接起來。在進行多重表格關聯查詢時,需要考慮表格之間的關係,並使用適當的關聯方法。以下介紹幾種多

如何使用PHP框架Yii開發一個高可用的雲端備份系統 如何使用PHP框架Yii開發一個高可用的雲端備份系統 Jun 27, 2023 am 09:04 AM

隨著雲端運算技術的不斷發展,資料的備份已經成為了每個企業必須要做的事情。在這樣的背景下,開發一款高可用的雲端備份系統尤其重要。而PHP框架Yii是一款功能強大的框架,可以幫助開發者快速建立高效能的Web應用程式。以下將介紹如何使用Yii框架開發一款高可用的雲端備份系統。設計資料庫模型在Yii框架中,資料庫模型是非常重要的一環。因為資料備份系統需要用到很多的表和關

php如何使用Yii3框架? php如何使用Yii3框架? May 31, 2023 pm 10:42 PM

隨著互聯網的不斷發展,Web應用程式開發的需求也越來越高。對於開發人員而言,開發應用程式需要一個穩定、高效、強大的框架,這樣可以提高開發效率。 Yii是一款領先的高效能PHP框架,它提供了豐富的特性和良好的性能。 Yii3是Yii框架的下一代版本,它在Yii2的基礎上進一步優化了效能和程式碼品質。在這篇文章中,我們將介紹如何使用Yii3框架來開發PHP應用程式。

Symfony vs Yii2:哪個框架比較適合開發大型Web應用? Symfony vs Yii2:哪個框架比較適合開發大型Web應用? Jun 19, 2023 am 10:57 AM

隨著Web應用需求的不斷增長,開發者在選擇開發框架方面也越來越有選擇的空間。 Symfony和Yii2是兩個備受歡迎的PHP框架,它們都具有強大的功能和效能,但在面對需要開發大型網路應用程式時,哪個框架更適合呢?接下來我們將對Symphony和Yii2進行比較分析,以幫助你更好地進行選擇。基本概述Symphony是一個由PHP編寫的開源Web應用框架,它是建立

Yii2 vs Phalcon:哪個框架更適合開發顯示卡渲染應用? Yii2 vs Phalcon:哪個框架更適合開發顯示卡渲染應用? Jun 19, 2023 am 08:09 AM

在當前資訊時代,大數據、人工智慧、雲端運算等技術已成為了各大企業關注的熱點。在這些技術中,顯示卡渲染技術作為一種高效能圖形處理技術,受到了越來越多的關注。顯示卡渲染技術廣泛應用於遊戲開發、影視特效、工程建模等領域。而對於開發者來說,選擇一個適合自己專案的框架,是一個非常重要的決策。在目前的語言中,PHP是一種相當有活力的語言,一些優秀的PHP框架如Yii2、Ph

Yii框架中的資料查詢:有效率地存取數據 Yii框架中的資料查詢:有效率地存取數據 Jun 21, 2023 am 11:22 AM

Yii框架是一個開源的PHPWeb應用程式框架,提供了眾多的工具和元件,簡化了Web應用程式開發的流程,其中資料查詢是其中一個重要的元件之一。在Yii框架中,我們可以使用類似SQL的語法來存取資料庫,從而有效率地查詢和操作資料。 Yii框架的查詢建構器主要包括以下幾種類型:ActiveRecord查詢、QueryBuilder查詢、命令查詢和原始SQL查詢

如何在 Golang 中使用關聯查詢連線表? 如何在 Golang 中使用關聯查詢連線表? Jun 05, 2024 am 11:10 AM

關聯查詢連接表透過SQL查詢結合多個表中的資料。在Golang中,使用sql.DB.Query()函數,指定查詢字串和參數。可以使用不同的關聯類型,如INNERJOIN、LEFTJOIN、RIGHTJOIN和FULLOUTERJOIN,取決於所需的結果集。

See all articles