news 2026/6/2 22:48:03

PHP 值对象实战指南:避免原始类型偏执

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
PHP 值对象实战指南:避免原始类型偏执

PHP 值对象实战指南:避免原始类型偏执

上一篇文章里,我们聊了原始类型偏执(Primitive Obsession)在 PHP 里为什么这么常见:邮箱、金额、日期、ID……统统用 string/int/float/array 传来传去。领域含义被抹平,校验逻辑散落在各处,代码越写越难改。

这一篇我们继续往下走:值对象(Value Object)不仅能让代码更清晰,还能让协作、测试和后续演进都更省心。不管你用的是 Laravel、Symfony 还是别的框架,只要项目里有明确的领域概念,值对象都能派上用场。

原文链接 PHP 值对象实战指南:避免原始类型偏执

1. 重复出现的模式:不只是代码味道

在 PHP 项目里,某些规则会反复出现。日期就是典型例子:

publicfunctionregisterEvent(string$eventDate,string$timeZone):void{if(!preg_match('/^\d{4}-\d{2}-\d{2}$/',$eventDate)){thrownewInvalidArgumentException('Invalid date format.');}if(!in_array($timeZone,DateTimeZone::listIdentifiers())){thrownewInvalidArgumentException('Invalid time zone.');}// More logic here...}

这里的日期和时区都用 string 表示。问题在于:一旦日期格式要改、或者时区规则有变化,你就会开始在各个角落复制粘贴同样的验证逻辑——漏一处就出事故。

2. 值对象登场:Date 和 TimeZone

与其在每个入口都手写校验,不如把“日期”“时区”做成值对象,让它们自己保证合法性。

2.1 Date 值对象

finalclassDate{privatestring$value;privatefunction__construct(string$date){if(!preg_match('/^\d{4}-\d{2}-\d{2}$/',$date)){thrownewInvalidArgumentException('Invalid date format.');}$this->value=$date;}publicstaticfunctionfromString(string$date):self{returnnewself($date);}publicfunctionvalue():string{return$this->value;}publicfunction__toString():string{return$this->value;}}

2.2 TimeZone 值对象

finalclassTimeZone{privatestring$value;privatefunction__construct(string$timeZone){if(!in_array($timeZone,DateTimeZone::listIdentifiers())){thrownewInvalidArgumentException('Invalid time zone.');}$this->value=$timeZone;}publicstaticfunctionfromString(string$timeZone):self{returnnewself($timeZone);}publicfunctionvalue():string{return$this->value;}publicfunction__toString():string{return$this->value;}}

2.3 在业务里怎么用

publicfunctionregisterEvent(Date$eventDate,TimeZone$timeZone):void{// No need for repetitive validation// Logic continues...}

参数一眼就能看懂,而且验证逻辑只存在一份:在值对象里。

3. 给值对象加上行为

值对象不只是“更强的类型”。它还能承载和这个概念紧密相关的行为。

比如你需要判断活动日期是否在未来,可以把逻辑放进 Date:

finalclassDate{// ..publicfunctionisInTheFuture():bool{$now=newDateTime();$eventDate=newDateTime($this->value);return$eventDate>$now;}}

这样就不用在每个用到日期的地方都重复写一遍比较逻辑,也更符合领域表达:判断未来与否,本来就是“日期”这个概念的一部分。

4. 金额:用值对象守住精度

处理金额是原始类型偏执最容易踩坑的地方之一。用 float 表示钱,舍入误差迟早会找上门;再加上币种、汇率,复杂度会迅速拉高。

把金额做成值对象,通常的做法是:

  • 用最小单位(比如分)把金额存成 int
  • 币种作为字段和金额绑定在一起

下面这个 Money 示例进一步加上了换汇的行为:

finalclassMoney{privateint$amount;// Stored in minor units (e.g., cents)privatestring$currency;// Constructor and other methods...publicfunctionconvertToCurrency(string$targetCurrency,float$exchangeRate):self{$convertedAmount=(int)round($this->amount*$exchangeRate);returnnewself($convertedAmount,$targetCurrency);}}

把规则关在 Money 里,你的业务代码就不用到处关心“这里是分还是元”“币种对不对”“舍入怎么做”。

5. 在框架里落地

一旦你开始用值对象,Laravel / Symfony 反而会更好用:你能把“原始数据 ↔ 值对象”的转换放到框架扩展点里,业务层拿到的就都是领域类型。

5.1 Laravel 示例

Laravel 里可以用自定义 cast,把数据库字段自动转成值对象:

useIlluminate\Contracts\Database\Eloquent\CastsAttributes;classMoneyCastimplementsCastsAttributes{publicfunctionget($model,string$key,$value,array$attributes){returnMoney::fromInt($value,'USD');}publicfunctionset($model,string$key,$value,array$attributes){return$valueinstanceofMoney?$value->amount():$value;}}

把这个 cast 挂到 Eloquent 模型上后,取出来的就是 Money,而不是裸值,代码会干净很多。

5.2 Symfony 示例

Symfony 里可以用 Doctrine 的 embeddables 或自定义 DBAL type,把 Money / EmailAddress 这类复杂类型映射到数据库:

classMoneyTypeextends\Doctrine\DBAL\Types\Type{constMONEY='money';// Custom type namepublicfunctionconvertToPHPValue($value,\Doctrine\DBAL\Platforms\AbstractPlatform$platform){returnMoney::fromInt($value,'USD');}publicfunctionconvertToDatabaseValue($value,\Doctrine\DBAL\Platforms\AbstractPlatform$platform){return$valueinstanceofMoney?$value->amount():$value;}}

这样做的好处是:从数据库到领域层,你始终在用“领域类型”,而不是一堆无意义的 string/int/float。

6. 怎么迁移现有代码库

引入值对象不需要推倒重来。最稳妥的方式是渐进式迁移:

  • 先改边界层:在 HTTP controller、表单请求、CLI 命令里,把输入的原始值解析成值对象
  • 再改服务层:逐步把 service 方法签名从原始类型换成值对象
  • 配合静态分析:用 PHPStan 或 Psalm 强化类型约束,尽早发现不匹配

7. 结语:让领域自己说话

值对象的意义,不在于“OO 更纯粹”,而在于让领域概念变得清楚、可约束、可复用。

下次你准备在方法里传一堆 string/int 的时候,不妨停一下问自己:

“这真的是一个简单值吗?还是一个应该被命名、被约束的领域概念?”

做出这个小改变,短期能减少重复校验和隐性 bug;长期则会让整个代码库更稳、更好改——也更照顾未来维护它的你。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/2 2:07:48

零代码打造全功能后端 API 的 JSON 传输协议

APIJSON APIJSON 是一个功能强大的 JSON 传输协议和 ORM 库,它允许前端(客户端)定制返回 JSON 的数据和结构,而后端无需编写任何代码即可提供 API 接口和文档。该项目由腾讯开源,已成为腾讯内部使用广泛的开源项目之一…

作者头像 李华
网站建设 2026/6/2 2:02:40

31、多线程编程全解析

多线程编程全解析 1. 多线程编程基础 多线程编程在现代软件开发中扮演着至关重要的角色。线程编程的接口是 POSIX 线程 API(通常称为 pthreads),它最初在 1995 年的 IEEE POSIX 1003.1c 标准中定义,作为 C 库 libpthread.so 的一部分实现。过去 15 年左右,pthreads 有…

作者头像 李华
网站建设 2026/5/31 1:34:11

Archipack建筑建模插件深度解析:从入门到精通的完整指南

Archipack建筑建模插件深度解析:从入门到精通的完整指南 【免费下载链接】archipack Archipack for blender 2.79 项目地址: https://gitcode.com/gh_mirrors/ar/archipack 在当今建筑设计与3D建模领域,Archipack作为Blender的专业插件&#xff0…

作者头像 李华
网站建设 2026/6/2 1:09:36

EmotiVoice在博物馆导览系统中的智能化升级应用

EmotiVoice在博物馆导览系统中的智能化升级应用 在一座安静的博物馆展厅里,一位游客驻足于一件距今三千年的青铜器前。耳边响起的声音不再是千篇一律、机械单调的讲解,而是一位沉稳如考古学家般的男声,语气中带着敬畏与发现的喜悦&#xff1a…

作者头像 李华
网站建设 2026/5/31 21:08:19

EmotiVoice语音合成在医疗陪护机器人中的应用前景

EmotiVoice语音合成在医疗陪护机器人中的应用前景 在一间安静的养老院房间里,一位年近八旬的老人正坐在轮椅上。他的子女常年在国外工作,平日里陪伴他的只有一台外形温和、带屏幕的护理机器人。当系统检测到他连续两小时未起身活动时,机器人缓…

作者头像 李华
网站建设 2026/6/2 7:33:07

屏幕录制新选择:vokoscreenNG如何解决你的录制难题

屏幕录制新选择:vokoscreenNG如何解决你的录制难题 【免费下载链接】vokoscreenNG vokoscreenNG is a powerful screencast creator in many languages to record the screen, an area or a window (Linux only). Recording of audio from multiple sources is supp…

作者头像 李华