3分钟短文:说说Laravel模型中还算常用的2个“关系”

简介: 只说简单的,不说复杂的

引言

上一章我们介绍了比较简单的laravel模型关联关系中的一对一,介绍了关联操作方法。
太难的概念理解起来都费劲,更不用说写代码了,所以对于太难的那些关联关系,
且不论其效率如何,我们都不先做介绍。

img

本期说一说2个比较常用的关联模型。

belongsTo 关系

正好像对于一个词语,找到对应的反义词,或者说有一个图片,找到其镜像图片这样的。
有作用力,就有反作用力。一对一关系模型中,A有一个B,则反过来,B属于一个A。
这就是首先要介绍的 belongsTo 关系。

在模型Profile中添加对应到User模型的关系:

class Profile extends Model {
    public function user()
    {
        return $this->belongsTo('App\User');
    }
}

也就是说,有一个profile是从属于user的,这与User模型的hasOne正好是对应关系。
在代码中使用该关联关系:

$email = Profile::where('id', 3)->first()->user->email;

其中first方法返回一个Profile模型对象实例,在Profile类中我们声明了 user() 方法用于关系用户模型,
所以此处链式调用 user 属性,返回的是一个 AppUser 对象实例,其包含 User 模型的所有属性,
因此 email 属性也相应返回数据库内的字段值。

一对多关系

还有一个常见的关联关系是一对多。比如一个用户有多个手机号,一种状态包含很多个事件,一个商品有多个标签等等等等,
这都是一对多的常见用法。

我们使用State模型状态有多个Event事件这个场景,演示一下一对多关系的声明,以及应用。在命令行创建模型文件,同时创建迁移文件:

php artisan make:model State --migration

默认在 AppState.php 文件内生成下面的代码:

use Illuminate\Database\Eloquent\Model;
class State extends Model {}

我们还是先去生成数据库表的迁移文件,手动实现迁移字段:

public function up()
{
    Schema::create('states', function(Blueprint $table)
    {
        $table->increments('id');
        $table->string('name');
        $table->string('abbreviation');
        $table->timestamps();
    });
}

以及撤回迁移时删除表:

public function down()
{
    Schema::drop('states');
}

接着在命令行执行迁移指令:

php artisan migrate

执行成功,数据库表states就创建成功了。

我们说关联关系需要外键,所以需要手动在events表内追加一个字段 state_id,用于指向刚才创建的表states的id字段。
执行命令行,创建迁移文件:

php artisan make:migration add_state_id_to_events_table --table=events

手动实现迁移文件的修改:

public function up()
{
    Schema::table('events', function (Blueprint $table) {
        $table->integer('state_id')->unsigned()->nullable();
        $table->foreign('state_id')->references('id')->on('states');
    });
}

以及回滚迁移时手动删除追加的字段:

public function down()
{
    Schema::table('events', function (Blueprint $table) {
        $table->dropForeign('events_state_id_foreign');
        $table->dropColumn('state_id');
    });
}

基础数据准备完毕,下面在模型内添加关联关系:

class State extends Model {
    public function events() {
        return $this->hasMany('App\Event');
    }
}

非常直观,一种状态,有若干个事件。反过来,一个事件,一定属于某种状态,那就是belongsTo关系。

class Event extends Model {
    public function state()
    {
        return $this->belongsTo('App\State');
    }
}

代码层面也准备好了,下面可以开始使用了。比如创建事件时,手动为其指定状态:

$event = new Event;
$event->name = "Laravel Hacking and Pizza";
$event->state_id = 41;
$event->save();

注意,hasMany关联关系,返回的是多个模型的集合,可以后续链式调用集合的所有方法。

写在最后

本文不失简单地介绍了belongsTo和hasMany两个关联关系,这在代码中仅次于hasOne关系,
使用的频次比较高的。而效率也就是根据外键多查询一次SQL的消耗而已。但是明白其中原理之后,
在代码内耗时的操作里,也绝不可滥用关联关系,否则会严重消耗性能。

Happy coding :-)

我是@程序员小助手,专注编程知识,圈子动态的IT领域原创作者

相关文章
|
7月前
|
机器学习/深度学习 人工智能 自然语言处理
【论文精读】AAAI 2022- 统一的命名实体识别作为词与词之间的关系分类
【论文精读】AAAI 2022- 统一的命名实体识别作为词与词之间的关系分类
【论文精读】AAAI 2022- 统一的命名实体识别作为词与词之间的关系分类
|
自然语言处理 算法 编译器
C++基础句法
● 使用场景 1.switch只能支持常量固定值相等的判断 2.if还可以判断区间范围 3.用switch能做的,用if都能做,但是反过来不行。
86 0
|
机器学习/深度学习 人工智能 自然语言处理
解决通用LLM「偏科」问题,数学大模型MathGPT要来了!
解决通用LLM「偏科」问题,数学大模型MathGPT要来了!
288 0
|
机器学习/深度学习 存储 数据采集
【英文文本分类实战】之二——数据集挑选与划分
【英文文本分类实战】之二——数据集挑选与划分
317 0
【英文文本分类实战】之二——数据集挑选与划分
【论文写作分析】之二 《基于类别混合嵌入的电力文本层次化分类方法》
【论文写作分析】之二 《基于类别混合嵌入的电力文本层次化分类方法》
【论文写作分析】之二 《基于类别混合嵌入的电力文本层次化分类方法》
|
机器学习/深度学习 算法
【论文写作分析】之五《融合类别特征扩展与N-gram子词过滤的fastText短文本分类》
【论文写作分析】之五《融合类别特征扩展与N-gram子词过滤的fastText短文本分类》
【论文写作分析】之五《融合类别特征扩展与N-gram子词过滤的fastText短文本分类》
|
机器学习/深度学习 自然语言处理
【BERT-多标签文本分类实战】之二——BERT的地位与名词术语解释
【BERT-多标签文本分类实战】之二——BERT的地位与名词术语解释
328 0
|
JSON 自然语言处理 数据处理
Pytext实战-构建一个文本分类器有多快
Pytext实战-构建一个文本分类器有多快
114 0
Pytext实战-构建一个文本分类器有多快
|
数据采集 人工智能 自然语言处理
谷歌提出多语言BERT模型:可为109种语言生成与语言无关的跨语言句子嵌入
谷歌研究人员提出了一种LaBSE的多语言BERT嵌入模型。该模型可为109种语言生成与语言无关的跨语言句子嵌入,同时在跨语言文本检索性能优于LASER。
402 0
谷歌提出多语言BERT模型:可为109种语言生成与语言无关的跨语言句子嵌入
|
机器学习/深度学习 人工智能 自然语言处理
潜心30年,知网知识系统如何从概念层次上计算自然语言
近日,机器之心采访了语知科技的首席科学家董强先生,董强向我们详细介绍了一种基于 Common-sense 知识库体系从概念层次进行自然语言处理的技术。语知自然语言理解技术平台正是基于知网语言知识库独有的语义分析技术,从概念层次上而不是从词的层面上进行自然语言处理,因此系统的计算复杂度会大大降低,也就更容易将系统离线部署到移动端或边缘设备中。深耕 30 多年 NLU 的知网(HowNet)如今正以语知科技创业公司的形象走进公众的视野。
265 0
潜心30年,知网知识系统如何从概念层次上计算自然语言