如果您已经了解 Java 中的 records,您可能会发现它的用法与类非常相似,但必须考虑到一些重要的差异。在本文中,我们将了解 Java 中记录和类之间的差异。如果您仍然不知道记录,我建议阅读我的文章《Java 中的记录:它们是什么以及如何使用它们》。
不可变对象是指一旦创建对象,其属性就无法修改的对象。对于 records 来说,它们是不可变的,也就是说,一旦创建 record 类型的对象,其属性就无法修改。另一方面,类可能是不变的,也可能不是不变的,具体取决于它的实现方式。这部分确保数据的完整性并防止其被意外修改。
类通常只是为了存储数据而编写,例如来自数据库查询的数据或来自表单的数据。在许多情况下,该数据是不可变的,因为需要在不使用同步的情况下确保数据的有效性。为了实现这一点,使用以下元素编写一个类:
例如,如果你有一个 Person 类,有两个属性 name 和 lastName,你可以这样写:
public class Person { private final String name; private final String lastName; public Person(String name, String lastName) { this.name = name; this.lastName = lastName; } public String getName() { return name; } public String getLastName() { return lastName; } @Override public String toString() { return "Person{" + "name='" + name + '\'' + ", lastName='" + lastName + '\'' + '}'; } @Override public boolean equals(Object o) { if (this == o) return true; if (!(o instanceof Person person)) return false; return Objects.equals(getName(), person.getName()) && Objects.equals(getLastName(), person.getLastName()); } @Override public int hashCode() { return Objects.hash(getName(), getLastName()); } }
这是该任务的解决方案,但它是实际需要的大量代码。如果类具有更多属性,即使在 IDE 或 GitHub Copilot 等插件的帮助下完成,编写的代码也会更长。更好的解决方案是将我们的类声明为数据类,即仅存储数据的类,并且不必具有特定行为,这就是 records 的用武之地。
这样,Person 类就可以重写为记录,如下所示:
public record Person(String name, String lastName) { }
这会自动生成 equals、hashCode 和 toString 方法,以及每个属性的 getter 方法。
如果需要的是一个不可变的数据结构来存储数据,并且不需要对属性进行修改(简单地看成是一个携带信息的对象)。另一方面,如果您需要一个具有独特逻辑和特定方法的更通用的结构、一种面向对象范例的方法、应用设计模式或使用 JPA 或 Hibernate 等,那么您应该使用类。
Consideremos el siguiente ejemplo, se tienen dos records Product con los atributos name y price, y Cart con un solo atributo products del tipo ArrayList
package org.jordi.example; public record Product(String name, double price) { }
package org.jordi.example; import java.util.ArrayList; import java.util.List; public record Cart(List<Product> products) { public Cart() { this(new ArrayList<>()); } public int getQuantity() { return this.products.size(); } public double getTotal() { return this.products.stream().mapToDouble(Product::price).sum(); } }
La cuestión en este caso es que cada uno de los record es inmutable por sí mismo, pero en el caso del record Cart al tener un atributo del tipo ArrayList<> y dado que por naturaleza un ArrayList es mutable, se puede modificar el contenido de la lista una vez que se crea una instancia del record Cart.
package org.jordi.example; public class Main { public static void main(String[] args) { Product water = new Product("Water", 15); Product milk = new Product("Milk", 22); Cart cart = new Cart(); cart.products().add(water); cart.products().add(milk); System.out.println("Price: " + cart.getTotal()); cart.products().clear(); System.out.println("Quantity: " + cart.getQuantity()); System.out.println("Price: " + cart.getTotal()); } }
El código anterior compila sin problemas, ya que solo se está modificando el contenido de la lista, pero no se está modificando el atributo products en sí. Este solo es un ejemplo para un caso particular, que probablemente no sea necesario, pero es bueno saber que esto se puede realizar.
The above is the detailed content of Records vs Classes in Java. For more information, please follow other related articles on the PHP Chinese website!