经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » 程序设计 » PHP » 查看文章
详解Laravel设置多态关系模型别名的方式
来源:jb51  时间:2019/10/18 8:32:58  对本文有异议

作为 Laravel 的重度使用者肯定都对多态关系不默生,以官方文档为例,文章有标签,视频有标签,那么文章和视频这些模型与标签模型的关系就是 多态多对多(Many To Many (Polymorphic))

如果我们给 ID 为 1 的文章打上两个标签,数据库标签关系表的的存储结果就是这样子:

  1. > select * from taggables;
  2. +--------+-------------+---------------+
  3. | tag_id | taggable_id | taggable_type |
  4. +--------+-------------+---------------+
  5. | 1 | 1 | App\Post |
  6. | 2 | 1 | App\Post |
  7. +--------+-------------+---------------+

相信有不少人和我一样希望 taggable_type 的值不要直接用模型类名,而是使用表名: posts 。官方文档的建议是:

  1. use Illuminate\Database\Eloquent\Relations\Relation;
  2.  
  3. Relation::morphMap([
  4. 'posts' => 'App\Post',
  5. 'videos' => 'App\Video',
  6. ]);

https://laravel.com/docs/6.x/eloquent-relationships#custom-polymorphic-types

我们可以将这个定义写到 AppServiceProvider 中,但是有一个非常严重的问题: 我们在新增或者删除模型的时候,会很容易忘记去更新这个定义 。我已经至少出现这个问题 3 次了,所以我一直在纠结有没有更好的方法,今天突然灵机一动,实现了一个看起来似乎是一个不错的方式,分享给大家。

思路来源

我尝试跟踪了一遍源码,发现模型中有一个方法 getMorphClass ,多态关联的时候,就是用它来取目标对象的类型名称的,默认返回类名:

  1. public function getMorphClass()
  2. {
  3. $morphMap = Relation::morphMap();
  4. if (! empty($morphMap) && in_array(static::class, $morphMap)) {
  5. return array_search(static::class, $morphMap, true);
  6. }
  7. return static::class;
  8. }

那么,只要我们在模型中覆盖这个方法便可以方便的实现目标了。

实现目标

我们有两个选择去实现它:

  1. 创建一个模型基类覆盖这个方法,所有的模型都来集成它即可;
  2. 创建一个 trait,在需要的模型中引入它。

我当然会选择 trait 方式来实现,不管从定义还是代码耦合度上,使用 trait 来解决这类特性需求都是再适合不过了,如果你对 trait 还不太熟悉,可以阅读我之前的文章: 《我所理解的 PHP Trait》

我们的目标是使用表名来做为关系类别名,那么在模型中如何获取表名呢,直接使用模型的 getTable 即可,那么整个 trait 的实现如下:

  1. app/Traits/UseTableNameAsMorphClass.php
  2. <?php
  3. namespace App\Traits;
  4. trait UseTableNameAsMorphClass
  5. {
  6. public function getMorphClass()
  7. {
  8. return $this->getTable();
  9. }
  10. }

然后在我们需要用到关系类型的模型中引入它即可:

  1. <?php
  2. namespace App;
  3. use App\Traits\UseTableNameAsMorphClass;
  4. use Illuminate\Database\Eloquent\Model;
  5. class Post extends Model
  6. {
  7. use UseTableNameAsMorphClass;
  8.  
  9. //...
  10. }

友情提示

当然,如果你习惯给表名加前缀,或者你的表名与模型名不太一致,那么,你只需要修改 trait 中 getMorphClass 的实现即可,我个人的习惯是模型名就是表名的单数,不带前缀。

总结

以上所述是小编给大家介绍的Laravel设置多态关系模型别名的方式,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对w3xue网站的支持!如果你觉得本文对你有帮助,欢迎转载,烦请注明出处,谢谢!

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号