文档章节

YII2中多表关联的使用

xsctx7788
 xsctx7788
发布于 2017/07/18 10:51
字数 921
阅读 9
收藏 0

首先先来说明一下表结构

表结构

现在有订单表、用户表、商品清单表、商品库存表,

订单表Order Id Order_id User_id  
用户表User Id User_id User_name  
商品清单表Order_goods Id Order_id Goods_id  
商品表Goods Id Goods_id Goods_name  
商品库存表Stock Id Stock-id Goods_id Stock_count

上页下页

 

 

 

在YII中,如果想直接关联其他表进行查询的话,需要先在模型里定义它们的关联

Order

class Order extends \yii\db\ActiveRecord.{

    // 关联函数以get+要关联的数据表名来命名

    // 这是获取下订单的客户

    public function getUser(){

        // 第一个参数为要关联的子表模型类名,

        // 第二个参数指定 通过子表的user_id,关联主表的usesr_id字段

        // 这里写清楚点大概意思就是User.user_id => Order.user_id

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

hasMany、hasOne使用

Yii2中的表之间的关联有2种,它们用来指定两个模型之间的关联。

  • 一对多:hasMany
  • 一对一:hasOne
  •  
  • 返回结果:这两个方法的返回结果都为yii\db\ActiveQuery对象(如果你想最后返回的是标准数组形式,记得加上asArray()参数)
  • 第一个参数:所关联的模型的类名称。
  • 第二个参数:是一个数组,其中键为所关联的模型中的属性,值为当前模型中的属性。

关联的使用

现在我们来尝试获取一个订单

//获取订单信息

$order = Order::findOne(1);

//根据订单信息获取到用户信息

$user = $order->user;

当然你可以选择使用with方法,这样看起来简洁一些,其中with的参数为关系的名称,也就在model里面定义的getUser中的user

//返回订单信息(包括用户信息)

$order = Order::find(1)->with('user');

//或者

$order = Order::find(1)->getUser();

上面的代码会生成并执行如下的sql语句

SELECT * FROM order    WHERE id=1;

SELECT * FROM user     WHERE user.user_id=order.user_id;

从上面可以看出访问一个关联的时候有两种方法

  • 如果以函数的方式调用,会返回一个 ActiveQuery 对象($customer->getOrders()->all())
  • 如果以属性的方式调用,会直接返回模型的结果($customer->orders)

关联结果缓存

如果这时order表发生了改变,我们希望再次查询的话

$user = $order->user;

再次得到订单的时候你会发现没有变化。原因是只会在第一次执行$order->user的时候才会去数据库里面查询,然后会把结果缓存起来,以后查询的时候都不会再执行sql。

那么如果你想再次执行sql如何做呢?可以执行

//先释放缓存

unset($order->user);

$order->user;

跨表查询

下面重点来了!通过上面表结构的图可以看到,User表和Order_goods表示没有直接关联的,那么如果我们想根据用户信息查找这个用户买了哪些商品的话,就势必需要通过Order表去关联两张表。那么该怎么做呢?首先还是model层。因为我们是根据用户去查,所以到User的model层去定义关联。

User

public function getOrder() {
    return $this->hasMany(Order::className(), ['user_id' => 'user_id']);
}

public function getOrderGoods() {
    return $this->hasMany(OrderGoods::className(), ['order_id' => 'order_id'])->
        via('order');
}

这里注意:getOrderGoods中的第二个order_id是指getOrder关联的Order中的order_id,第一个order_id是指OrderGoods中的order_id。

但是!我们还有最简单的方法,那就是使用SQL语句啦!

$map = 'select
       user.name,
       order.id,
       order_goods.goods_id,
       goods.goods_name,
       stock.stock_count
       from user
       LEFT JOIN order          ON order.user_id = user.user_id
       LEFT JOIN order_goods    ON order_goods.order_id = order.order_id
       LEFT JOIN goods          ON goods.goods_id = order_goods.goods_id
       LEFT JOIN stock          ON stock.goods_id = goods.goods_id';
$list1   = Article::findBySql($map)->asArray()->all();

这样基本就是整个关联部分了

© 著作权归作者所有

xsctx7788
粉丝 1
博文 29
码字总数 18797
作品 0
苏州
程序员
私信 提问
YII2.0 AcriveRecord的简单使用

这篇文章我们来看看在 Yii2 之中的 Active Record,为啥要将 Active Record 单独出来说呢?因为个人认为这是 Yii(不管是 Yii1.1 还是 Yii2)最强大的一部分功能之一,何况又遇上在 Yii2 中其...

bieru
2015/05/10
0
0
yii2 ActiveRecord多表关联以及多表关联搜索的实现

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

白狼栈
2016/07/05
54
0
Yii框架多表查询(一对一与多对一)

在网上查过了很多yii框架多表查询的问题,但总没有适合自己的,自己建了几张表过来分析,把操作写出来分享Yii框架多表查询(一对一与多对一)给大家 查一对一或多对一(常使用到的就是文章和文章分...

李佳顺
2014/04/10
3K
0
Yii2的ActiveRecord与ActiveQuery简单介绍

活动记录 活动记录为与数据库中某张表中的某条记录相关联的对象,我们通过调用此对象的 CURD方法对记录进行操作,Yii2的活动记录其实很标准,AR 类为模型Model,对应数据表,AR 类的实例为活...

big_cat
2016/04/18
2.8K
0
yii2组件之多图上传插件FileInput的详细使用

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

白狼栈
2016/06/23
158
0

没有更多内容

加载失败,请刷新页面

加载更多

OpenStack 简介和几种安装方式总结

OpenStack :是一个由NASA和Rackspace合作研发并发起的,以Apache许可证授权的自由软件和开放源代码项目。项目目标是提供实施简单、可大规模扩展、丰富、标准统一的云计算管理平台。OpenSta...

小海bug
昨天
5
0
DDD(五)

1、引言 之前学习了解了DDD中实体这一概念,那么接下来需要了解的就是值对象、唯一标识。值对象,值就是数字1、2、3,字符串“1”,“2”,“3”,值时对象的特征,对象是一个事物的具体描述...

MrYuZixian
昨天
6
0
数据库中间件MyCat

什么是MyCat? 查看官网的介绍是这样说的 一个彻底开源的,面向企业应用开发的大数据库集群 支持事务、ACID、可以替代MySQL的加强版数据库 一个可以视为MySQL集群的企业级数据库,用来替代昂贵...

沉浮_
昨天
6
0
解决Mac下VSCode打开zsh乱码

1.乱码问题 iTerm2终端使用Zsh,并且配置Zsh主题,该主题主题需要安装字体来支持箭头效果,在iTerm2中设置这个字体,但是VSCode里这个箭头还是显示乱码。 iTerm2展示如下: VSCode展示如下: 2...

HelloDeveloper
昨天
7
0
常用物流快递单号查询接口种类及对接方法

目前快递查询接口有两种方式可以对接,一是和顺丰、圆通、中通、天天、韵达、德邦这些快递公司一一对接接口,二是和快递鸟这样第三方集成接口一次性对接多家常用快递。第一种耗费时间长,但是...

程序的小猿
昨天
9
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部