前段时间比较忙,就没有坚持写博客,但发现这周末再想捡起来,好难,一直到了今天晚上,才决定坐下来写一篇,哈哈哈~~

最近在用 Laravel 5.2,踩了几个关于 Relation 的坑,在这里用博客记录一下。

如果大家不了解 Laravel 的 Relation 的话,可以查看文档Eloquent: Relationships

在数据库中先创建两张表 users 和 password_resets,分别代表用户和重置密码,假设他们的关系是一对多,一个用户有多个重置密码记录。

我们可以使用 Laravel 提供的 migration 创建数据库,具体可以查看文档Database: Migrations

使用的users 表和 password_resets 表是默认创建的。

然后我们可以通过界面注册一个用户,也可以在 users 表中直接插入一条。password_resets 表中也插入几条数据,这个也可以通过界面的忘记密码添加。

然后创建好相应的Model,在 User.php 中,加入如下代码:

public function passwordReset() {
return $this->hasMany('App\PasswordReset', 'email', 'email');
}

在 Controller 中使用如下代码:

$data = User::with('passwordReset')->get();
echo $data->toJson();
die;

这个时候你可以看到输出的数据中的 password_reset 字段是有内容的,但假如说你希望查找的 password_reset 中只有 token,写法如下:

$data = User::with(['passwordReset' => function($query) {
$query->select('token');
}])->get();

然后你会惊奇的发现输出的数据中的 password_reset 字段是一个空数组。

是不是觉得好神奇,当时我也是崩溃的,去查看源码,发现原来,Relation 是通过两条语句实现的,它会先查主表(users表),选出其中关联的字段(email字段)组成数组,使用 in 语句去查询关联表(password_resets表),然后将查出来的数据根据关联字段相等,处理到一起。这里需要注意的是查出来的两块数据关联到一起是在 php 中实现的,所以,上面的语句由于没有查出 password_resets表的 email 字段,无法关联到一起,其结果的 password_reset 字段是一个默认值-空数组。

现在我们在 select 中加入 email 字段,代码如下:

$data = User::with(['passwordReset' => function($query) {
$query->select('token', 'email');
}])->get();

然后你就会发现数据中的 password_reset 字段有值了。

这是踩到的一个坑。

然后我们在来设置一下 PasswordReset.php 中设置一下主键,设置为 email 字段(这里可能不太合理,email 会重复,这样写只是为了重现这个坑),代码如下:

protected $primaryKey = 'email';

然后你又会惊奇的发现输出的数据中的 password_reset 字段又变成了一个空数组,这又是怎么了?

原来 Laravel 5.2 为 Model 添加了一个属性 $keyType,默认值是 int,这意味着它会将查出来的 email 字段强转成 int 类型,所以在匹配的时候,就有对不上了。

现在我们只需要在定义一下 $keyType 为 string 就可以了,在 PasswordReset.php 中添加如下代码:

protected $keyType = 'string';

这是在 Laravel 5.0 升 5.2时踩到的坑。

Laravel 5.2,我也是刚开始看,如果上面有什么不对的,欢迎大家指出~~

如果大家觉得这篇博客对你有所帮助的话,请点一下顶~~

Laravel学习--关于Relation的坑的更多相关文章

  1. 后端PHP框架laravel学习踩的各种坑

    安装完laravel的ventor目录后出现“Whoops, looks like something went wrong.”这样的错误信息 打开config/app.php,打开debug为tru ...

  2. Laravel学习笔记(三)--在CentOS上配置Laravel

    在Laravel框架上开发了几天,不得不说,确实比较优雅,处理问题逻辑比较清楚.     今天打算在CentOS 7上配置一个Laravel,之前都是在本机上开发,打算实际配置一下.     1)系统 ...

  3. Laravel学习笔记之Session源码解析(上)

    说明:本文主要通过学习Laravel的session源码学习Laravel是如何设计session的,将自己的学习心得分享出来,希望对别人有所帮助.Laravel在web middleware中定义了 ...

  4. 《PHP框架Laravel学习》系列分享专栏

    <PHP框架Laravel学习>已整理成PDF文档,点击可直接下载至本地查阅https://www.webfalse.com/read/201735.html 文章 Laravel教程:l ...

  5. Laravel 学习 .env文件 getenv 获得环境变量的值

    Laravel 学习 .env文件 getenv 获得环境变量的值  我们还需要对应用的 .env 文件进行设置,为应用指定数据库名称 sample. .env . . . DB_DATABASE=s ...

  6. laravel学习之旅

    前言:之前写了二篇YII2.0的基本mvc操作,所以,打算laravel也来这一下 *安装现在一般都用composer安装,这里就不讲述了* 一.熟悉laravel (1)如果看到下面这个页面,就说明 ...

  7. laravel学习:主从读写分离配置的实现

    本篇文章给大家带来的内容是关于laravel学习:主从读写分离配置的实现,有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助. 在DB的连接工厂中找到以下代码.../vendor/larav ...

  8. Laravel学习笔记之PHP反射(Reflection) (上)

    Laravel学习笔记之PHP反射(Reflection) (上) laravel php reflect 2.1k 次阅读  ·  读完需要 80 分钟 3 说明:Laravel中经常使用PHP的反 ...

  9. Laravel chunk和chunkById的坑

    Laravel chunk和chunkById的坑 公司中的项目在逐渐的向Laravel框架进行迁移.在编写定时任务脚本的时候,用到了chunk和chunkById的API,记录一下踩到的坑. 一.前 ...

随机推荐

  1. [APUE]进程控制(上)

    一.进程标识 进程ID 0是调度进程,常常被称为交换进程(swapper).该进程并不执行任何磁盘上的程序--它是内核的一部分,因此也被称为系统进程.进程ID 1是init进程,在自举(bootstr ...

  2. TypeScript: Angular 2 的秘密武器(译)

    本文整理自Dan Wahlin在ng-conf上的talk.原视频地址: https://www.youtube.com/watch?v=e3djIqAGqZo 开场白 开场白主要分为三部分: 感谢了 ...

  3. Socket聊天程序——服务端

    写在前面: 昨天在博客记录自己抽空写的一个Socket聊天程序的初始设计,那是这个程序的整体设计,为了完整性,今天把服务端的设计细化记录一下,首页贴出Socket聊天程序的服务端大体设计图,如下图: ...

  4. 通过重建Hosting系统理解HTTP请求在ASP.NET Core管道中的处理流程[下]:管道是如何构建起来的?

    在<中篇>中,我们对管道的构成以及它对请求的处理流程进行了详细介绍,接下来我们需要了解的是这样一个管道是如何被构建起来的.总的来说,管道由一个服务器和一个HttpApplication构成 ...

  5. 用scikit-learn学习DBSCAN聚类

    在DBSCAN密度聚类算法中,我们对DBSCAN聚类算法的原理做了总结,本文就对如何用scikit-learn来学习DBSCAN聚类做一个总结,重点讲述参数的意义和需要调参的参数. 1. scikit ...

  6. MAC下 mysql不能插入中文和中文乱码的问题总结

    MAC下 mysql不能插入中文和中文乱码的问题总结 前言 本文中所提到的问题解决方案,都是基于mac环境下的,但其他环境,比如windows应该也适用. 问题描述 本文解决下边两个问题: 往mysq ...

  7. 自己来实现一个简易的OCR

    来做个简易的字符识别 ,既然是简易的 那么我们就不能用任何的第三方库 .啥谷歌的 tesseract-ocr, opencv 之类的 那些玩意是叼 至少图像处理 机器视觉这类课题对我这种高中没毕业的人 ...

  8. 初识的Spring Mvc-----原理

    一.Spring Mvc简介 Spring Mvc(Spring Web Mvc) 属于表现层的框架. 二.Spring结构图 Spring Mvc是Spring框架里面web模块的一部分,是在Spr ...

  9. 原生JavaScript实现hasClass、addClass、removeClass、toggleClass

    兼容IE6+,因IE6.IE7.IE8不支持Array.prototype.indexOf()和String.prototype.trim(),分别用Polyfill实现支持. 详细: indexOf ...

  10. atitit.管理学三大定律:彼得原理、墨菲定律、帕金森定律

    atitit.管理学三大定律:彼得原理.墨菲定律.帕金森定律 彼得原理(The Peter Principle) 1 彼得原理解决方案1 帕金森定律 2 如何理解墨菲定律2 彼得原理(The Pete ...