Yii2 ActiveRecord多表关联及多表关联搜索的实现


Posted in PHP onJune 30, 2016

Yii的ActiveRecord是与数据库打交道的类,也即MVC中的M(模型层),也是ORM的O(Object)。

一个老生常谈的问题。最近通过群里的反馈,觉得很多人还是没有去理解这个问题。今天把这个问题讲明白了,看看yii2 ActiveRecord是怎么个多表关联以及如何去优化这个关联。

场景需求:

假设我们有一张用户表user和一张用户渠道表auth,两张数据表通过user.id和auth.uid进行一对一关联。现需要在user列表展示auth表的来源渠道source,且该渠道可搜索。

首先我们先通过gii生成user和auth系列相关的model和操作。此处不做详细说明,有关gii的操作可参考xxx

我看继续看重要的几个操作步骤:

1、找到user表对应的AR模型类 common\models\User.php,在该类文件中进行关联auth表

/**
* 关联auth表
*/
public function getAuth()
{
// hasOne要求返回两个参数 第一个参数是关联表的类名 第二个参数是两张表的关联关系 
// 这里uid是auth表关联id, 关联user表的uid id是当前模型的主键id
return $this->hasOne(common\models\Auth::className(), ['uid' => 'id']);
}

设置好了之后,并不代表两张数据表自动进行关联了!我们访问user列表页(该列表页采用gii生成,目前我们没操作过),通过debug查看Database Queries不难发现,实际中的query并没有进行关联auth表

2、在gridview中添加关联表的来源渠道字段source

<?= GridView::widget([
// other codes
'columns' => [
// other columns
'auth.source',
]
]); ?>

有同学感觉疑问了,上面不是说了没进行关联吗,这个怎么可以直接使用auth.source?

先别急,此时我们打开debug看看实际的query。

我们会发现有很多类似 select * from `auth` where uid = xxx;之类的操作,如果你的分页默认20条数据时,会有20个类似的query。

我们先搞明白发生了什么?

实际上这属于php的基础知识了。读取和写入对象的一个不存在的成员变量时, __get() __set() 魔术函数会被自动调用。yii也是利用了这一点对其进行了实现!

该操作跟大部分人在gridview中封装方法获取关联表数据几乎一致,但是!20条sql的查询明显增加了众多的开销。如果这里是left join操作多好!

3、优化sql

我们需要优化的是:

20条sql变1条sql

只获取关联表需要的字段

有同学要嚷嚷了,这里是yii自带的操作,怎么优化?我们回到数据源的获取上,发现user列表的数据是通过userSearch model的search方法提供的。

也就是说我们的数据查询实际上就没有去进行关联表查询!既然如此,我们就在UserSearch加上关联查询

$query = User::find();
$query->joinWith(['auth']);
$query->select("user.*, auth.source");

我们再来刷新下user列表页,然后通过debug分析发现有两条sql引起了我们的注意

SELECT `user`.*, `auth`.`source` FROM `user` LEFT JOIN `auth` ON `user`.`id` = `auth`.`uid` LIMIT 20
SELECT * FROM `auth` WHERE `user_id` IN (20个uid);

也就是说我么已经达到了优化sql的目的,通过debug分析发现,DB的查询时间少了很多。

4、关联表字段增加查询

gridview中的搜索模型也是通过searchModel实现的,该模型通过rules控制着哪个字段可搜索,哪个字段不可搜索。

我们现在需要增加关联表的source可搜索,因此我们在searchModel中定义一个属性source且添加到rules中

public $source;
public function rules()
{
return [
// other rules
['source', 'safe'],
];
}

接着我们把gridview中的auth.source修改一下

// 'auth.source',
[
'attribute' => 'source',
'value' => 'auth.source',
'label' => '渠道来源',
],

到这里我们界面上是ok的,要实现程序上的搜索还差一步,我们在数据源获取的地方加上新增的source条件即可

$query->andFilterWhere([
// other params
'auth.source' => $this->source,
]);

下面给大家补充yii中ActiveRecord的一些用法

1,对象转数组

$model = new ActiveRecord();
$model.toArray();

由于ActiveRecord不是简单数组,不能直接json_encode,否则信息不完整。

解决办法:$model.toArray();这样就变为简单数组了,可以进行json_encode了。

2,通过名字或其他字段直接获取ActiveRecord的id。

$nIdcId = idc_info::model()->find('name like :name',array(':name'=>"%".$strIdcName."%"))->id;

我以前经常使用的办法是(现在发现很土):

$idc = Idc::model()->find("..."); 
$id = $idc->id;

3,对model的理解

$accModel = call_user_func(array(ActiveRecordName, 'model')); 
$model  = $accModel->findByPk($id);

以上所述是小编给大家介绍的Yii2 ActiveRecord多表关联及多表关联搜索的实现的相关知识,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!

PHP 相关文章推荐
在php中使用sockets:从新闻组中获取文章
Oct 09 PHP
PHP新手上路(十一)
Oct 09 PHP
PHP完整的日历类(CLASS)
Nov 27 PHP
PHP语法速查表
Jan 02 PHP
PHP中的错误处理、异常处理机制分析
May 07 PHP
浅谈php中mysql与mysqli的区别分析
Jun 10 PHP
PHP中array_merge和array相加的区别分析
Jun 17 PHP
PHP 二维数组根据某个字段排序的具体实现
Jun 03 PHP
php出现web系统多域名登录失败的解决方法
Sep 30 PHP
PHP判断是否为空的几个函数对比
Apr 21 PHP
php检测图片主要颜色的方法
Jul 01 PHP
PHPMailer发送邮件
Dec 28 PHP
eclipse php wamp配置教程
Jun 30 #PHP
PHP上传图片类显示缩略图功能
Jun 30 #PHP
PHP使用php-resque库配合Redis实现MQ消息队列的教程
Jun 29 #PHP
Thinkphp批量更新数据的方法汇总
Jun 29 #PHP
ThinkPHP实现更新数据实例详解(demo)
Jun 29 #PHP
php结合mysql与mysqli扩展处理事务的方法
Jun 29 #PHP
php简单解析mysqli查询结果的方法(2种方法)
Jun 29 #PHP
You might like
怎么样可以把 phpinfo()屏蔽掉?
2006/11/24 PHP
网页游戏开发入门教程二(游戏模式+系统)
2009/11/02 PHP
翻译整理的jQuery使用查询手册
2007/03/07 Javascript
提高javascript效率 一次判断,而不要次次判断
2012/03/30 Javascript
利用JavaScript脚本实现滚屏效果的方法
2015/07/07 Javascript
JS判断页面是否出现滚动条的方法
2015/07/17 Javascript
jQuery事件委托之Safari
2016/07/05 Javascript
详解js的事件处理函数和动态创建html标记方法
2016/12/16 Javascript
js控制按钮,防止频繁点击响应的实例
2017/02/15 Javascript
基于jQuery的左滑出现删除按钮的示例
2017/08/29 jQuery
js实现鼠标移动到图片产生遮罩效果
2017/10/21 Javascript
原生javascript运动函数的封装示例【匀速、抛物线、多属性的运动等】
2020/02/23 Javascript
Element Dropdown下拉菜单的使用方法
2020/07/26 Javascript
vue实现tab栏点击高亮效果
2020/08/19 Javascript
[57:18]DOTA2上海特级锦标赛主赛事日 - 1 败者组第一轮#3VP VS VG
2016/03/03 DOTA
python计算程序开始到程序结束的运行时间和程序运行的CPU时间
2013/11/28 Python
python实现从网络下载文件并获得文件大小及类型的方法
2015/04/28 Python
详谈Python3 操作系统与路径 模块(os / os.path / pathlib)
2018/04/26 Python
Python饼状图的绘制实例
2019/01/15 Python
详解安装mitmproxy以及遇到的坑和简单用法
2019/01/21 Python
numpy中三维数组中加入元素后的位置详解
2019/11/28 Python
Python 矩阵转置的几种方法小结
2019/12/02 Python
使用OpenCV获取图片连通域数量,并用不同颜色标记函
2020/06/04 Python
5个你不知道的HTML5的接口介绍
2013/08/07 HTML / CSS
WoolOvers爱尔兰:羊绒、羊毛和棉针织品
2017/01/04 全球购物
Booking.com荷兰:全球酒店网上预订
2017/08/22 全球购物
医学毕业生自荐信
2013/10/11 职场文书
大学生优秀自荐信范文
2014/02/25 职场文书
汽车促销活动方案
2014/03/31 职场文书
档案保密承诺书
2014/06/03 职场文书
学校2014重阳节活动策划方案
2014/09/16 职场文书
2014年银行信贷员工作总结
2014/12/08 职场文书
金秋助学感谢信
2015/01/21 职场文书
退休欢送会主持词
2015/07/01 职场文书
幼儿园教师心得体会范文
2016/01/21 职场文书
oracle通过存储过程上传list保存功能
2021/05/12 Oracle