Java 8 Streams API 有助于以函数式而非命令式的方式编写代码。正如我们所知,不可变性提供了许多好处,因此,我尝试在可行的情况下使对象不可变。在日常编程中,我发现自己处于想要“设置”一个值的情况。虽然我的对象是不可变的,所以我需要创建一个新对象并在构造函数中初始化该字段。
我使用项目 Lombok,它提供诸如 @Value
之类的注解,这基本上使对象不可变。它还具有 @Builder
,它使用构建器模式为不可变对象(immutable对象)提供构建器,将 Fluent API 中未提及的字段设置为 null
。
@Builder
注释有一个名为 toBuilder
的字段,当设置为 true 时,它提供 toBuilder()
方法,该方法返回一个构建器,其中填充了对象的字段,开发人员可以在其中“设置”值、调用 build()
并返回一个新对象。
例如要创建一个包含修改过的 forename
字段的不可变对象(immutable对象)的 List
,我会执行以下操作:
import java.util.Arrays;
import java.util.List;
import java.util.stream.Collectors;
import lombok.Builder;
import lombok.Value;
public class SOExample {
@Value
@Builder(toBuilder = true)
private static class Person {
private final String forename;
private final String surname;
private final int age;
private final int heightInCm;
}
public static void main(String[] args) {
List<Person> people = Arrays.asList(Person.builder()
.forename("stack")
.surname("overflow")
.age(21)
.heightInCm(180)
.build());
people.stream()
.map(p -> p.toBuilder()
.forename("updatedForename")
.build())
.collect(Collectors.toList());
}
}
如果不使用 Lombok,这将需要大量样板代码。事实上,我查看了生成的代码,发现它既不小也不琐碎。这让我质疑自己。其他人是怎么做的?我担心我错过了一个技巧。
考虑到上面的描述和示例,返回具有更新字段的不可变对象(immutable对象)的副本的最佳方法是什么?如何在 Streams API 中使用它?
最佳答案
您发现自己不得不这样做的频率如何?
如果您要代替拥有可修改的类,那么这可能表明真正的不变性对您的类型没有意义。
如果您没有经常这样做,那么仅使用原始访问器(和对值的修改)生成一个新访问器似乎也不错;您展示的 Builder
示例也没有。
Person personA = ...
Person personB = new Person("updated", personA.getSurname(), personA.getAge(), ...);
关于java - 在处理不可变对象(immutable对象)时,如何最好地返回带有修改字段的副本?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46125552/