是否允许在构造时修改值对象的值

问题描述

假设我希望以下值对象包含始终大写的字符串值。在构造函数中使用 toupperCase() 是否有资格这样做?

class CapitalizedId(value: String) {
    val value: String = value.toupperCase()
    // getters
    // equals and hashCode
}

解决方法

一般来说,我认为在值对象的构造函数中执行如此简单的转换没有问题。对于构造函数的用户来说,当然应该不会感到意外,但正如名称 CapitalizedId 已经告诉您,无论将创建什么都将大写,从我的角度来看,这并不奇怪。我还在构造函数中执行有效性检查,以确保遵守业务不变量。

如果您担心不在构造函数中执行操作,或者如果操作和验证变得过于复杂,您可以随时提供工厂方法(或在 Kotlin 中使用 companion,我猜,不是 Kotlin 专家)包含所有繁重的工作(想想 LocalDateTime.of())和验证逻辑,然后像这样使用它:

CapitalizedId.of("abc5464g"); 

注意:在这种情况下,在实现工厂方法时,构造函数应该是私有的

,

在构造函数中使用 toUpperCase() 是否有资格这样做?

是的,从某种意义上说,您最终得到的仍然是 ValueObject 模式的表达式。

这与初始化器应该初始化并且不包括其他职责的想法不一致。见Misko Hevery 2008

这个特定的实现会是一个代价高昂的错误吗?应该不会