详解YII关联查询


Posted in PHP onJanuary 10, 2016

一、多表关联的配置

在我们使用 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 相关文章推荐
用定制的PHP应用程序来获取Web服务器的状态信息
Oct 09 PHP
追求程序速度,而不是编程的速度
Apr 23 PHP
全世界最小的php网页木马一枚 附PHP木马的防范方法
Oct 09 PHP
php fsockopen解决办法 php实现多线程
Jan 20 PHP
PHP yii实现model添加默认值的方法(两种方法)
Nov 10 PHP
Yii2中添加全局函数的方法分析
May 04 PHP
php 猴子摘桃的算法
Jun 20 PHP
Yii2框架中使用PHPExcel导出Excel文件的示例
Aug 09 PHP
PHP操作Redis数据库常用方法示例
Aug 25 PHP
PHP DB 数据库连接类定义与用法示例
Mar 11 PHP
Laravel获取当前请求的控制器和方法以及中间件的例子
Oct 11 PHP
PHP设计模式之组合模式定义与应用示例
Feb 01 PHP
PHP 设计模式系列之 specification规格模式
Jan 10 #PHP
PHP生成各种常见验证码和Ajax验证过程
Jan 10 #PHP
PHP常用字符串操作函数实例总结(trim、nl2br、addcslashes、uudecode、md5等)
Jan 09 #PHP
PHP统计目录中文件以及目录中目录大小的方法
Jan 09 #PHP
PHP基于单例模式实现的mysql类
Jan 09 #PHP
thinkPHP查询方式小结
Jan 09 #PHP
thinkPHP中多维数组的遍历方法
Jan 09 #PHP
You might like
php 正则 过滤html 的超链接
2009/06/02 PHP
PHP simple_html_dom.php+正则 采集文章代码
2009/12/24 PHP
PHP字符串函数系列之nl2br(),在字符串中的每个新行 (\n) 之前插入 HTML 换行符br
2011/11/10 PHP
Yii2创建控制器(createController)方法详解
2016/07/23 PHP
php实现用户注册密码的crypt加密
2017/06/08 PHP
laravel使用Faker数据填充的实现方法
2019/04/12 PHP
JavaScript 实现模态对话框 源代码大全
2009/05/02 Javascript
Javascript 遍历对象中的子对象
2009/07/03 Javascript
JavaScript实用技巧(一)
2010/08/16 Javascript
jquery批量设置属性readonly和disabled的方法
2014/01/24 Javascript
JavaScript获取图片的原始尺寸以宽度为例
2014/05/04 Javascript
Javascript基础教程之switch语句
2015/01/18 Javascript
AngularJs基于角色的前端访问控制的实现
2016/11/07 Javascript
jquery结合html实现中英文页面切换
2016/11/29 Javascript
微信开发之调起摄像头、本地展示图片、上传下载图片实例
2016/12/08 Javascript
微信小程序实战之登录页面制作(5)
2020/03/30 Javascript
AngularJS实现注册表单验证功能
2017/10/16 Javascript
详解vue几种主动刷新的方法总结
2019/02/19 Javascript
Fundebug支持监控微信小程序HTTP请求错误的方法
2019/02/21 Javascript
继承行为在 ES5 与 ES6 中的区别详解
2019/12/24 Javascript
从零开始用webpack构建一个vue3.0项目工程的实现
2020/09/24 Javascript
Vue使用鼠标在Canvas上绘制矩形
2020/12/24 Vue.js
基于vue+echarts数据可视化大屏展示的实现
2020/12/25 Vue.js
[26:24]完美副总裁、DOTA2负责人蔡玮专访:电竞如人生
2014/09/11 DOTA
Python数据类型中的“冒号“[::]——分片与步长操作示例
2018/01/24 Python
Python字典中的键映射多个值的方法(列表或者集合)
2018/10/17 Python
python粘包问题及socket套接字编程详解
2019/06/29 Python
Python退出时强制运行一段代码的实现方法
2020/04/29 Python
canvas 如何绘制线段的实现方法
2018/07/12 HTML / CSS
PHP开发工程师面试问题集锦
2012/11/01 面试题
大学生农村教师实习自我鉴定
2013/09/21 职场文书
党员年终民主评议的自我评价
2013/11/05 职场文书
最新优秀教师个人先进事迹材料
2014/05/06 职场文书
先进集体事迹材料范文
2014/12/25 职场文书
2015年污水处理厂工作总结
2015/05/26 职场文书
Python pandas读取CSV文件的注意事项(适合新手)
2021/06/20 Python