PHP编程:PHP laravel中的多对多关系实例详解

《PHP编程:PHP laravel中的多对多关系实例详解》要点:
本文介绍了PHP编程:PHP laravel中的多对多关系实例详解,希望对您有用。如果有疑问,可以联系我们。

PHP实例数据表之间是纵横交叉、相互关联的,laravel的一对一,一对多比较好理解,官网介绍滴很详细了,在此我就不赘述啦,重点我记下多对多的关系

PHP实例一种常见的关联关系是多对多,即表A的某条记录通过中间表C与表B的多条记录关联,反之亦然.比如一个用户有多种角色,反之一个角色对应多个用户.

PHP实例为了测试该关联关系,我们沿用官网的用户角色示例:

PHP实例需要三张数据表:users、roles 和 role_user,role_user 表按照关联模型名的字母顺序命名(这里role_user是中间表),并且包含 user_id 和 role_id两个列.

PHP实例多对多关联通过编写返回 belongsToMany 方法返回结果的方法来定义.废话不说多,直接上数据结构:

PHP实例1:创建一个角色表roles,并添加一些初始化数据:

PHP实例
SET FOREIGN_KEY_CHECKS=0;
-- ----------------------------
-- Table structure for users
-- ----------------------------
DROP TABLE IF EXISTS `users`;
CREATE TABLE `users` (
`id` int(10) unsigned NOT NULL AUTO_INCREMENT,`name` varchar(255) COLLATE utf8_unicode_ci NOT NULL,`email` varchar(255) COLLATE utf8_unicode_ci NOT NULL,`password` varchar(60) COLLATE utf8_unicode_ci NOT NULL,`remember_token` varchar(100) COLLATE utf8_unicode_ci DEFAULT NULL,`created_at` timestamp NOT NULL DEFAULT '0000-00-00 00:00:00',`updated_at` timestamp NOT NULL DEFAULT '0000-00-00 00:00:00',PRIMARY KEY (`id`),UNIQUE KEY `users_email_unique` (`email`) USING BTREE
) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci;
-- ----------------------------
-- Records of users
-- ----------------------------
INSERT INTO `users` VALUES ('1','admin','admin@163.com','$2y$10$J/yXqscucanrHAGZp9G6..Tu1Md.SOljX3M8WrHsUdrgat4zeSuhC','ilocXtjZJwhrmIdLG1cKOYegeCwQCkuyx1pYAOLuzY2PpScQFT5Ss7lBCi7i','2016-04-21 16:26:23','2016-12-14 09:29:59');
INSERT INTO `users` VALUES ('2','baidu','10940370@qq.com','$2y$10$2A5zJ4pnJ5uCp1DN3NX.5uj/Ap7P6O4nP2BaA55aFra8/rti1K6I2',null,'2016-04-22 06:48:10','2016-04-22 06:48:10');
INSERT INTO `users` VALUES ('3','fantasy','1009@qq.com','','2017-06-14 10:38:57','2017-06-15 10:39:01');

PHP实例2:创建一个角色表roles,并添加一些初始化数据:

PHP实例
SET FOREIGN_KEY_CHECKS=0;
-- ----------------------------
-- Table structure for roles
-- ----------------------------
DROP TABLE IF EXISTS `roles`;
CREATE TABLE `roles` (
`id` int(10) unsigned NOT NULL AUTO_INCREMENT,PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=7 DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci;
-- ----------------------------
-- Records of roles
-- ----------------------------
INSERT INTO `roles` VALUES ('1','超级版主','2016-12-14 09:29:59');
INSERT INTO `roles` VALUES ('2','司令','2016-04-22 06:48:10');
INSERT INTO `roles` VALUES ('3','军长','2017-06-15 10:39:01');
INSERT INTO `roles` VALUES ('4','司长','2017-06-07 10:41:41','2017-06-15 10:41:51');
INSERT INTO `roles` VALUES ('5','团战','2017-06-22 10:41:44','2017-06-28 10:41:54');
INSERT INTO `roles` VALUES ('6','小兵','2017-06-22 10:41:47','2017-06-22 10:41:56');

PHP实例3:创建一个中间表role_user用于记录users表与roles表的对应关系,并添加一些初始化数据:

PHP实例
SET FOREIGN_KEY_CHECKS=0;
-- ----------------------------
-- Table structure for role_user
-- ----------------------------
DROP TABLE IF EXISTS `role_user`;
CREATE TABLE `role_user` (
 `id` int(10) unsigned NOT NULL AUTO_INCREMENT,`user_id` int(11) DEFAULT NULL,`role_id` int(11) DEFAULT NULL,`created_at` datetime DEFAULT NULL,`updated_at` datetime DEFAULT NULL,PRIMARY KEY (`id`)
) ENGINE=MyISAM AUTO_INCREMENT=8 DEFAULT CHARSET=latin1;
-- ----------------------------
-- Records of role_user
-- ----------------------------
INSERT INTO `role_user` VALUES ('1','1','2','2017-06-07 11:42:13','2017-06-21 11:32:16');
INSERT INTO `role_user` VALUES ('2','3','2017-06-07 11:32:13','2017-06-07 11:22:13');
INSERT INTO `role_user` VALUES ('3','4','2017-06-07 11:12:13');
INSERT INTO `role_user` VALUES ('4','5','2017-06-07 11:22:13');
INSERT INTO `role_user` VALUES ('5','6','2017-06-07 11:52:13');
INSERT INTO `role_user` VALUES ('6','2017-06-07 11:42:13');
INSERT INTO `role_user` VALUES ('7','2017-06-07 11:52:13');

PHP实例注意我们定义中间表的时候没有在结尾加s并且命名规则是按照字母表顺序,将role放在前面,user放在后面,并且用_分隔,这一切都是为了适应Eloquent模型关联的默认设置:在定义多对多关联的时候如果没有指定中间表,Eloquent默认的中间表使用这种规则拼接出来.

PHP实例创建一个Role模型:

PHP实例
<?php
namespace App\Models;
use Illuminate\Database\Eloquent\Model;
/**
 * Class Role
 * @package App\Models
 * @mixin \Eloquent
 */
class Role extends Model
{
}

PHP实例然后我们在 User 模型上定义 roles 方法:

PHP实例
<?php
namespace App\Models;
use Illuminate\Database\Eloquent\Model;
/**
 * Class User
 * @package App\Models
 * @mixin \Eloquent
 */
class User extends Model
{
  /**
   * 用户角色
   */
  public function roles()
  {
    return $this->belongsToMany('App\Models\Role');
  }
}

PHP实例注:正如我们上面提到的,如果中间表不是role_user,那么需要将中间表作为第二个参数传入belongsToMany方法,如果中间表中的字段不是user_id和role_id,这里我们姑且将其命名为$user_id和$role_id,那么需要将$user_id作为第三个参数传入该方法,$role_id作为第四个参数传入该方法,如果关联方法名不是roles还可以将对应的关联方法名作为第五个参数传入该方法.

PHP实例接下来我们在控制器中编写测试代码:

PHP实例
<?php
$user = User::find(1);
$roles = $user->roles;
echo '用户'.$user->name.'所拥有的角色:';
foreach($roles as $role)
  echo $role->name.' '; //对应输出为:用户admin所拥有的角色:司令 军长 团战 

PHP实例当然,和所有其它关联关系类型一样,你可以调用roles 方法来添加条件约束到关联查询上:

PHP实例
User::find(1)->roles()->orderBy('name')->get();

PHP实例正如前面所提到的,为了确定关联关系连接表的表名,Eloquent 以字母顺序连接两个关联模型的名字.不过,你可以重写这种约定 ―― 通过传递第二个参数到 belongsToMany 方法:

PHP实例
return $this->belongsToMany('App\Models\Role','user_roles');

PHP实例除了自定义连接表的表名,你还可以通过传递额外参数到 belongsToMany 方法来自定义该表中字段的列名.第三个参数是你定义关联关系模型的外键名称,第四个参数你要连接到的模型的外键名称:

PHP实例
return $this->belongsToMany('App\Models\Role','user_roles','user_id','role_id');

PHP实例定义相对的关联关系

PHP实例要定义与多对多关联相对的关联关系,只需在关联模型中调用一下 belongsToMany 方法即可.我们在 Role 模型中定义 users 方法:

PHP实例
<?php
namespace App\Models;
use Illuminate\Database\Eloquent\Model;
/**
 * Class Role
 * @package App\Models
 * @mixin \Eloquent
 */
class Role extends Model
{
  /**
   * 角色用户
   */
  public function users()
  {
    return $this->belongsToMany('App\Models\User');
  }
}

PHP实例正如你所看到的,定义的关联关系和与其对应的User 中定义的一模一样,只是前者引用 App\Models\Role,后者引用App\Models\User,由于我们再次使用了 belongsToMany 方法,所有的常用表和键自定义选项在定义与多对多相对的关联关系时都是可用的.

PHP实例测试代码如下:

PHP实例
$role = Role::find(2);
$users = $role->users;
echo '角色#'.$role->name.'下面的用户:';
foreach ($users as $user) 
  echo $user->name.' ';//对应输出为:角色#司令下面的用户:admin fantasy baidu

PHP实例正如你看到的,处理多对多关联要求一个中间表.Eloquent 提供了一些有用的方法来与这个中间表进行交互,例如,我们假设 User 对象有很多与之关联的 Role 对象,访问这些关联关系之后,我们可以使用这些模型上的pivot 属性访问中间表字段:

PHP实例
$roles = User::find(1)->roles;
foreach ($roles as $role) 
  echo $role->pivot->role_id.'<br>';//对应输出为:2 3 5

PHP实例注意我们获取到的每一个 Role 模型都被自动赋上了 pivot 属性.该属性包含一个代表中间表的模型,并且可以像其它 Eloquent 模型一样使用.

dawei

【声明】:唐山站长网内容转载自互联网,其相关言论仅代表作者个人观点绝非权威,不代表本站立场。如您发现内容存在版权问题,请提交相关链接至邮箱:bqsm@foxmail.com,我们将及时予以处理。