Laravel Model 利用 Macroable 为数据模型添加宏能力

2017-08-23 15:55:56 +08:00
 Zhiyicx

[摘要] 简单的说一下宏能力,这个类是 Illuminate\Support\Traits\Macroable 其中利用重载实现了可以定义宏的功能,即通过 macro 静态方法添加回调,并定义一个名字。利用 __call 当前类没有这个函数的时候执行这个函数名注册的回调。

产生需求

在使用 Laravel 开发 ThinkSNS Plus 的时候,因为很多功能块都没有写在一个库里面,利用拓展包的形式添加实际功能,里面很多地方也用到了“多态多对多”的关系。问题来了,开发一个问答程序,想要给用户模型增加发布的问题或者回答的关系,起初是继承一份 User 模型,添加了关系,之后就发现问题了,因为用户的 tag 是使用多态多对多的关系,我通过继承的用户模型是无法拿到这种关系数据的因为 ***able_type 是 user 数据模型类名称或者别名。而我继承之后类也就发生改变了。

完成需求

随之想到,在 Laravel 中有一个 Trait 叫做 Macroable 然后发现 Builder 都有这种能力,而 Model 没有,随之也将这个 Trait 添加到要使用的 model 上,后来发现,如果其他模型也要用是不是也要再添加一次?随之写了一份 Trait:

只要在要使用的 model 中 use 即可。

使用

有了这个 Trait 那么我们添加到 User 模型中,就可以使用宏能力为其动态添加函数了:

这样,我们可以直接 :

拿到用户发布的所有问题了。

以上代码都来自于 ThinkSNS Plus,看完整的开发代码可以看仓库:

GitHub: https://github.com/slimkit/thinksns-plus(开源不易,求 Star )

ThinkSNS 官网:http://thinksns.com/

如果还有任何疑问,请咨询 QQ3298713109

1783 次点击
所在节点    程序员
2 条回复
Zhiyicx
2017-08-23 16:45:47 +08:00
随之写了一份 Trait:

trait Macroable
{
use \Illuminate\Support\Traits\Macroable {
__call as macroCall;
}

/**
* Get a relationship value from a method.
*
* @param string $key
* @return mixed
* @author Seven Du <shiweidu@outlook.com>
*/
public function getRelationValue($key)
{
$relation = parent::getRelationValue($key);
if (! $relation && static::hasMacro($key)) {
return $this->getRelationshipFromMethod($key);
}

return $relation;
}

/**
* Handle dynamic method calls into the model.
*
* @param string $method
* @param array $parameters
* @return mixed
* @author Seven Du <shiweidu@outlook.com>
*/
public function __call($method, $parameters)
{
if (static::hasMacro($method)) {
return $this->macroCall($method, $parameters);
}

return parent::__call($method, $parameters);
}

/**
* Handle dynamic static method calls into the method.
*
* @param string $method
* @param array $parameters
* @return mixed
*/
public static function __callStatic($method, $parameters)
{
return parent::__callStatic($method, $parameters);
}
}
Zhiyicx
2017-08-23 16:46:10 +08:00
使用

有了这个 Trait 那么我们添加到 User 模型中,就可以使用宏能力为其动态添加函数了:

User::macro('questions', function () {
return $this->hasMany(Question::class, 'user_id', 'id');
});

这样,我们可以直接 :

$questions = $user->questions;

拿到用户发布的所有问题了。

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/385201

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX