【laravel-amount】Model 中自动转换金额的 Trait

背景

系统中涉及到金额的字段,View 层表现的时候一般都是以为单位使用小数形式展示,不过 Domain 层存储时从空间、性能、容错角度出发,经常以为单位,用整型来存储。

在 Lavarel 中,可以在 Model 中添加属性方法进行转换

public function getAmountAttribute($value)
{
    return $value / 100;
}

public function setAmountAttribute($value)
{
    $this->attributes['amount'] = (int)($value * 100);
}

不过涉及金额的字段比较多时就需要定义很多相同逻辑的函数,本项目即将该逻辑抽出为 Trait,简化金额字段相关的处理。

原理

将转换逻辑封装在 AmountTrait 中,覆写 Model 类的 getMutatedAttributes, mutateAttributeForArray, getAttributeValue 及 setAttribute 方法,当访问相关字段时自动进行转换处理。

public static $amountTimes = 100;

public function getMutatedAttributes()
{
    $attributes = parent::getMutatedAttributes();
    return array_merge($attributes, $this->getAmountFields());
}

protected function mutateAttributeForArray($key, $value)
{
    return (in_array($key, $this->getAmountFields()))
        ? $value / self::$amountTimes
        : parent::mutateAttributeForArray($key, $value);
}

public function getAttributeValue($key)
{
    $value = parent::getAttributeValue($key);
    if (in_array($key, $this->getAmountFields())) {
        $value = $value / self::$amountTimes;
    }

    return $value;
}

public function setAttribute($key, $value)
{
    if (in_array($key, $this->getAmountFields())) {
        $value = (int)($value * self::$amountTimes);
    }
    parent::setAttribute($key, $value);
}

public function getAmountFields()
{
    return (property_exists($this, 'amountFields')) ? $this->amountFields : [];
}

依赖

Laravel >= 5.2

安装

composer require "hao-li/laravel-amount:dev-master"

使用

  1. 在 Model 中引用 AmountTrait
use HaoLi\LaravelAmount\Traits\AmountTrait;
  1. 使用 AmountTrait
use AmountTrait;
  1. 定义金额字段(本例中为 amount)
protected $amountFields = ['amount'];
  1. 完成
    之后读取 amount 字段时,该字段的内容会自动从数据库的转换为,向其赋值时反之从转换为

FAQ

和别的 trait 中方法冲突

以 setRawAttributes 为例(此为之前方案,目前并未覆写此方法,仅为举例,其他方法原理相同)

  1. 将冲突的方法分别重命名
use AmountTrait, BTrait {
    AmountTrait::setRawAttributes as amountTraitSetRawAttributes;
    BTrait::setRawAttributes as BTraitSetRawAttributes;
}
  1. 在 Model 中定义该冲突的方法,根据情况分别调用别名方法
public function setRawAttributes(array $attributes, $sync = false)
{
    $this->BTraitSetRawAttributes($attributes, $sync);
    $attributes = $this->getAttributes();
    $this->amountTraitSetRawAttributes($attributes, $sync);
}

注意这里 $attributes 可能已被改变,所以再次使用时要重新取得最新值

Github

hao-li/laravel-amount

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,478评论 19 139
  • 缘起 后端开发的基本操作就是处理数据 -- "增删改查 / CURD", 而 Laravel 框架的"对象关系映射...
    patiencing阅读 13,619评论 0 6
  • 用理工科思维看待这个世界 系列爬虫专栏 初学者,尽力实现最小化学习系统 主题:Scrapy 实战,并分别存储在My...
    谢小路阅读 4,357评论 0 5
  • 努力生活,不论怎样 怕的是你有一颗想扎在泥潭里的心 怕的是你以为会慢慢变好 一下子全部变回了原来的样子 我好像一直...
    drama1919阅读 2,087评论 0 0
  • 年末——多谢相伴
    春儿sweet阅读 1,365评论 0 0