java - 在处理不可变对象(immutable对象)时,如何最好地返回带有修改字段的副本?

标签 java java-8 java-stream immutability lombok

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/

相关文章:

java - 基于 User-Agent 的 Change Velocity 模板

java - 如何在 java 8 中使用流过滤两个列表对象并将值设置为新列表

java - java Stream.peek() 如何影响字节码?

Java流类型问题

java - 是否可以在方法定义中使用普通的 BiFunction

java - 使用Java编写的RESTFul api登录/注册

java - 禁用选项卡滑动

java - 需要澄清java垃圾收集复制收集器

java - 是否可以在 Java 8 中将日志记录添加到 removeIf?

java - Java 中的数值规范