使用不可变对象时,如何最好地返回带有修改字段的副本?

Rob*_*ain 5 java immutability lombok java-8 java-stream

Java 8 Streams API 有助于编写函数式代码,而不是命令式代码。正如我们所知,不变性提供了许多好处,因此,我尝试在任何可行的地方使对象不可变。在日常编程中,我发现自己想“设置”一个值。我的对象是不可变的,所以我需要创建一个新对象并在构造函数中初始化该字段。

我使用 Lombok 项目,它提供注释,例如@Value本质上使对象不可变。它还具有@Builderwhich 使用构建器模式为不可变对象提供构建器,将 fluent API 中未提及的字段设置为null.

@Builder注解有一个名为 的字段toBuilder,当设置为 true 时,它​​提供toBuilder()方法,该方法返回一个填充有来自对象的字段的构建器,开发人员可以在其中“设置”值、调用build()和返回一个新对象。

EG 创建一个List具有修改forename字段的不可变对象,我会执行以下操作:

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());
    }
}
Run Code Online (Sandbox Code Playgroud)

如果不使用 Lombok,这将需要大量样板代码。事实上,我已经查看了生成的代码,它既不小也不琐碎。这让我开始质疑自己。外面的人是怎么做的?我担心我错过了一个技巧。

考虑到上面的描述和示例,返回具有更新字段的不可变对象副本的最佳方法是什么,以及它将如何在 Streams API 中使用?

And*_*eas 5

您可以按照新的 Java 8 Time API 的方式进行操作,通过添加withXxx()方法,例如LocalDate

  • withYear(int year)LocalDate- 返回更改年份的副本。

  • withMonth(int month)LocalDate- 返回一份更改了月份的副本。

  • withDayOfMonth(int dayOfMonth)LocalDate- 返回一份更改了月份的副本。


use*_*398 4

您多久发现自己不得不这样做?

如果您这样做而不是拥有可修改的类,那么这可能表明真正的不变性对您的类型没有意义。

如果您没有定期执行这一切,那么仅使用原始访问器(以及对值的修改)生成一个新的访问器似乎并不算太​​糟糕;你所展示的例子也没有Builder

Person personA = ...

Person personB = new Person("updated", personA.getSurname(), personA.getAge(), ...);
Run Code Online (Sandbox Code Playgroud)