数据库+模型 - AR模型联表查询

  • 作者:KK

  • 发表日期:2017.01.24


使用joinWith实现左联接

用模型进行联表查询之前最好先给模型定义好模型关系,假设Article模型的getUser这个getter是通过hasOne返回文章的发表用户模型,则要一次性用join方法查出来是这样:

//SELECT `article`.* FROM `article` LEFT JOIN `user` ON `article`.`user_id` = `user`.`id` WHERE `article`.`id`=1
$article = Article::find()
	->where(['article.id' => 1])
	->joinWith('user')
	->orderBy(['user.id' => SORT_DESC])
	->one();

echo $article->title;
echo $article->user->name; //不会产生查询,因为上面join的时候已经查过了,所以user模型已经生成

其中要注意where的条件要声明表名以区分条件字段,否则容易造成同名字段冲突


使用innerJoinWith实现内联接

查询所有文章以及作者

//SELECT * FROM `article` INNER JOIN `user` ON `article`.`user_id` = `user`.`id` WHERE `article`.`id` IN (1,2,3)

$articles = Article::find()->innerJoinWith('user')->where(['in', 'article.id', [1, 2, 3]])->all();

foreach($articles as $article){
	echo $article->user->name; //不会产生查询
}

指定表别名

Article::find()->joinWith(['user u'])->orderBy(['u.id' => SORT_DESC])……

多表联接

/*
SELECT * FROM article 
LEFT JOIN user on article.user_id = user.id 
LEFT JOIN article_type ON article.type_id = article_type.id ……
*/
Article::find()->joinWith(['user', 'articleType'])……

这样就会调用Article::getUser和Article::getArticleType两个方法来获取关系并构造联表了

联表方法本来是可以接收一个字符串作为快速联单表的,如果要联多表,就像上面这样写成数组,然后一个数组元素就是一个联表,你要联多少表就堆多少个元素吧


join的时候在on条件里追加更多条件

以上例子中都只是单纯地拿两个表的关系字段做on对等比较,如果有自己自定义需求可以用onCondition方法:

//SELECT `article`.* FROM `article` LEFT JOIN `user` ON (`article`.`user_id` = `user`.`id`) AND (`user`.`status` = 1) WHERE `article`.`id`=1
$article = Article::find()
	->where(['article.id' => 1])
	->joinWith([
		'user' => function($arQuery){
			$arQuery->onCondition(['user.status' => 1]); //重点
		}
	])
	->one();

echo $article->title;
echo $article->user->name; //不会产生查询,因为上面join的时候已经查过了,所以user模型已经生成

这个条件追加与AR模型查询优化里面的在with时追加其它条件小节的做法是一样的,把joinWith或innerJoinWith里的参数换成数组,key是关联属性名称(对应的模型getter),value是一个回调,接收一个AR查询器,通过这个AR查询器用onCondition来追加条件

但不能用这样的链式查询来实现:joinWith('user')->onCondition($condition)->all()这是无效的代码

其中onCondition方法的条件声明和where是一样的