这里有新鲜出炉的精品教程,程序狗速度看过来!
Yii 2 完全根据 Yii 1.1 版本重写,后者是最流行的 PHP 框架之一。Yii 2 继承了 Yii 的简洁、快速、和高扩展性。Yii 2 需要 PHP 5.4,并且拥有现代 web 应用开发中最好的实例和协议。
这篇文章主要介绍了 Yii2 ActiveRecord 多表关联及多表关联搜索的实现的相关资料, 需要的朋友可以参考下
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);
来源: http://www.phperz.com/article/17/0731/343418.html