1.什么叫多态关系?
英文叫polymorphic, 本意是一个物体可以有很多的状态。在laravel里面,就是指各各模型直接有很多相互的关系,laravel有3种多态关系。我们举个例子,一个图片的模型可以归属于一个用户,也可以归属于一个文章,一篇文章可以有多个图,一个用户也可以有多个图。如何在模型文件里反应这些关系呢?我们可以用一个图片模型来反应这个关系。下面我们通过一个例子来说明这个关系。
2.创建文章模型,图片模型,然后我们来写一个迁移文件和工厂文件来填充假数据到文章数据库。
php artisan make:model Post -mf
在文章里,我们填两个字段:
$table->string('title');
$table->text('body');
然后我们创建一个图片模型文件:
php artisan make:model Image -m
我们暂时不需要创建工厂文件了。
在填迁移文件的时候我们需要填两个字段:
在通常情况下,我们会填一个post_id来表明这个图片是属于这个POST的。
但是在多态情况下,这种做法是无法反应用户ID和这个图片的关系。
所以在这种情况下我们就加一个特殊的字段,叫做"imageable_id".
还有一个字段是填入我们的模型,一般的做法是加一个"imageable_type"这个字段。
让我们来看下Image这个模型的迁移文件里加了什么:
$table->string('image_url');
$table->unsignedBigInteger('imageable_id');
$table->string('imageable_type');
第一个字段我们填的是上传后的图片连接。
第二个字段里我们填的是模型的ID,例如如果是post的话就是post_id, 如果是user的话是user_id.
第三个字段我们填的是模型类型:如App\Post或者App\User等。
然后我们来迁移。我们来填充一个假数据到post这个表格。
3.一对一的模型文件里多态关系的写法:
在文章模型文件里,我们可以写:
app/Post.php里面
public function image()
{
return $this->morphOne(Image::class, 'imageable');
}
app/Image.php里面:
在这里我们必须要用imageable这个方法名,因为我们在上面已经定义了。
public function imageable()
{
return $this->morphTo();
}
就好比我们belongTo()一样,这里要用morphTo().
下面我们来填充一个数据到图片这个模型里。
Post::first()->image()->create(['image_url' => 'some/image.jpg']);
我们可以用关系来填充,如果我们看下数据库里面的话,我们会看到:
imageable_id是1,imageable_type是App\Post.
接下来我们要写User.php里面的来填充一个假数据并且加一个图片到对应关系里面的表格。
在User.php里面的关系的语句跟post里面的是一模一样的。我们还是用同样的方法来填充数据:
User::first()->image()->create(['image_url' => 'some/image.jpg']);
数据库里表现的是:imageable_id是1,imageable_type是App\User.
4. 一对多的多态关系:
一篇文章可以有多个评论,一个用户可以有多个评论。
创建模型文件:
php artisan make:model Comment
迁移文件里面:
$table->text('body');
$table->unsignedBigInteger('commentable_id');
$table->string('commentable_type');
在Post里面:
public function comments()
{
return $this->morphMany(Comment::class, 'commentable');
}
在Comment里面,加关系方法:
public function commentable()
{
return $this->morphTo();
}
我们利用这个关系来添加一个评论:
Post::first()->comments()->create(['body' => 'my message']);
我们会看到数据库里多一个记录。
imageable_id是1,imageable_type是App\Post.
当然因为它是1对多的关系,我们可以继续加一条记录。
用同样的方法,我们可以加一个评论,用User的模型。
5.多对多的多态关系
多对多的多态关系,跟上面的不同点比较多,我们来用tags(标签)来做下例子。任何模型都可以有很多tag。标签也属于很多的模型。例如一个帖子post有很多标签,php, java, python等等,反过来也一样。
我们首先需要创建一个模型,然后需要再创建一个“胶水”,用这个“胶水”把标签模型和其他的模型联系到一起。
php artisan make:model tag -m
我们再做一个“胶水”的迁移文件:
php artisan make:migration creates_taggables_table --create taggables
我们把这个迁移文件打开进行修改。
$table->increments('id');
我们将这句改一下,因为我们要把tag模型和taggables直接做个关联。
$table->unsignedBigInteger('tag_id');
然后再加2个字段:
$table->unsignedBigInteger('taggable_id');
$table->string('taggable_type');
在tag这个表格上我们只需要加一个字段
$table->string('name');
这样我们所要做的工作就结束了,我们在Post这个模型上加一个方法:
Post.php
public function tags()
{
return $this->morphToMany(Tag::class, 'taggable');
}
在tags.php里面,我们可以加
public function posts()
{
return $this->morphedByMany(Post::class, 'taggable');
}
比较好的记忆方法是,在要被关联的模型里加一个以表格名称起名的方法,然后跟模型和关键词('taggable')进行关联,然后在另一个模型里放morphedByMany这个关键词,放对应的模型和关键词就行可以了。
如果我们试着放一个数据到这里的时候,我们会发现:
Post::first()->tag()->create(['name' => 'php']);
在taggables表格里多出了一个记录,而且在tags表格里多出了一个记录。
网友评论