白狼栈 2016-07-05 19:52:39 40963次浏览 19条评论 41 9 0

作者:白狼 出处:http://www.manks.top/yii2-many-ar-relation_search.html 本文版权归作者,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。

一个老生常谈的问题。最近通过群里的反馈,觉得很多人还是没有去理解这个问题。今天把这个问题讲明白了,看看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,
]);

[考虑目前国内网站大部分采集文章十分频繁,更有甚者不注明原文出处,原作者更希望看客们查看原文,以防有任何问题不能更新所有文章,避免误导!]

查看原文

觉得很赞
  • 评论于 2016-08-28 16:48 举报

    这篇文章应该写进教科书!

    3 条回复
    评论于 2016-08-29 09:20 回复

    谢谢支持!

    评论于 2018-09-20 11:50 回复

    二笔 😆

    评论于 2019-07-16 22:45 回复

    严重同意,支持

    觉得很赞
  • 评论于 2016-08-29 16:23 举报

    及时雨,简单明了,谢谢楼主分享

  • 评论于 2016-09-02 17:16 举报

    牛逼!!!!!!!!!!

  • 评论于 2016-09-03 09:22 举报

    粉楼主很久了,现在公司用TP了 不过还是很喜欢YII ,准备自己用YII,搞点东西了,支持楼主 加油。。。。。。。。

  • 评论于 2016-09-27 14:41 举报

    按楼主的方法怎么不行啊?报错

    Invalid Parameter – yii\base\InvalidParamException
    app\modules\ptdt\models\PdVisitInfoh has no relation named "pd_visit_infol".
    ↵
    Caused by: Unknown Method – yii\base\UnknownMethodException
    Calling unknown method: app\modules\ptdt\models\PdVisitInfoh::getpd_visit_infol()
    
    in F:\guowencong\phpStudy\WWW\php_project\fjj_erp\vendor\yiisoft\yii2\base\Component.php at line 285
    
    1 条回复
    评论于 2017-03-30 15:05 回复

    你的解决了吗,我遇到这个问题

  • 评论于 2016-10-01 11:55 举报

    LZ高手,写的好几篇文章都很简洁实用

  • 评论于 2016-11-29 10:30 举报

    高手!!!

  • 评论于 2016-12-21 22:24 举报

    auth是表名 还是关系名容易搞混,比如有表前缀的时候会表名和关系名不同,能做出说明就好了,$query->joinWith(['auth'])和Grideview中使用的是关系名,其他地方使用的是表名,我觉得这些地方说清楚会更好

    1 条回复
    评论于 2016-12-28 15:27 回复

    我的表有前缀,弄混了,试了几次才好

    觉得很赞
  • 评论于 2016-12-29 21:41 举报

    试了好多次都报错

  • 评论于 2016-12-29 21:41 举报

    试了好多次都报错

  • 评论于 2017-02-28 08:13 举报

    不适合初学者 这个是白狼大神的通病 大神只需要把代码共享出来 自会有人分析

    1 条回复
    评论于 2018-12-25 15:29 回复

    同感。虽然很感谢白狼栈大神。但是表示看不懂。我的情况是A hasMany B,现在想用AB表的某些字段来筛选查询A和B的记录。不知道这样是否可以称为关联查询。
    不过我找到了yii2官网的教程,感觉还不错。https://www.yiiframework.com/wiki/780/drills-search-by-a-hasmany-relation-in-yii-2-0

    , , 觉得很赞
  • 评论于 2017-04-17 11:48 举报

    A跟B表关联,B跟C表关联,但是有个需求是,A里面有B的数据,但是他还行看到C的数据,怎么搞,

    return $this->hasMany(Pet::className(), ['user_id' => 'id']);

    怎么加才能出来,

  • 评论于 2017-06-16 14:23 举报

    有大婶在吗?我联表了,但是查不出信息啊,关联表信息不显示

  • 评论于 2017-08-09 11:03 举报

    文件中searchModel指的是什么或者说是哪个文件,在yii2中哪个位置?

  • 评论于 2018-01-18 14:22 举报

    打印出来的SQL语句是对的,但是前端显示却是not set空的

            $query2 = ServerAccount::find();
            $query2->joinWith(['accountType']);
            $query2->select('server_account.*,account_type.type');    
            $b = clone $query2;
            $lastSql = $b->createCommand()->getRawSql();
            print_r($lastSql); //SELECT `server_account`.*, `account_type`.`type` FROM `server_account` LEFT JOIN `account_type` ON `server_account`.`account_type` = `account_type`.`id` 
            $dataProvider = new ActiveDataProvider([
                'query' => $query2,
                'pagination' => [
                    'pageSize' => 10,
                ],
            ]);
            print_r($query2->all());//如下
    

    微信截图_20180118141727.png

    2 条回复
    评论于 2018-01-18 14:26 回复
                [id] => 11
                    [account_type] => 2
                    [name] => a
                    [hostname] => q,x,a,才,1
                    [ip] => 1
                    [username] => 123
                    [password] => dsadsa
                    [expiry] => 20010101
                    [created_at] => 1515465533
                    [updated_at] => 1515468022
                )
    
            [_related:yii\db\BaseActiveRecord:private] => Array
                (
                    [accountType] => common\models\AccountType Object
                        (
                            [_attributes:yii\db\BaseActiveRecord:private] => Array
                                (
                                    [id] => 2
                                    [type] => 服务器账号
                                )
    
                            [_oldAttributes:yii\db\BaseActiveRecord:private] => Array
                                (
                                    [id] => 2
                                    [type] => 服务器账号
                                )
    
    评论于 2018-01-18 14:27 回复

    account_type的type字段没有加到数组里

  • 评论于 2018-05-23 16:57 举报

    为什么我第二部就失败了
    Unknown Property – yii\base\UnknownPropertyException
    Getting unknown property: app\modules\member\models\LeagueMemberInfo::user_info

    1 条回复
    评论于 2018-05-24 14:02 回复

    找到问题了 原来关系名是小驼峰命名

  • 评论于 2019-01-17 11:50 举报

    QQ截图20190117114348.png
    QQ截图20190117114358.png
    为啥我在查询的时候 还是报没有找到goods 错误啊

  • 评论于 2021-11-05 14:15 举报

    真几把难用 laralve > tp > Yii

您需要登录后才可以评论。登录 | 立即注册