Si ya conoces los records en Java puede que su uso te resulte muy similar al de las clases, pero hay diferencias importantes que se deben tener en consideración. En este artículo vamos a ver las diferencias entre los records y las clases en Java. Si aún no conoces los records te recomiendo la lectura de mi post Records en Java: qué son y cómo utilizarlos.
Un objeto inmutable es aquel cuyos atributos no pueden ser modificados una vez que se ha creado el objeto. En el caso de los records, estos son inmutables, es decir, una vez que se ha creado un objeto de tipo record, no se pueden modificar sus atributos. Por otra parte, una clase puede o no ser inmutable, dependiendo de cómo se haya implementado. Esta parte asegura la integridad de los datos y evita que se modifiquen de forma accidental.
Comúnmente se escriben clases simplemente para almacenar datos, como los de una consulta a una base de datos, o los datos de un formulario. En muchos casos, estos datos son inmutables, ya que se necesita asegurar la validez de los datos sin utilizar sincronización. Para lograr esto se escribe una clase con los siguientes elementos:
Por ejemplo, si se tiene una clase Person con dos atributos name y lastName, se podría escribir de la siguiente manera:
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()); } }
Esta es una solución para la tarea, pero es mucho código para lo que realmente se necesita. En caso de que la clase tuviera más atributos, sería un código aún más largo de escribir, incluso si se realiza con la ayuda de un IDE o un plugin como GitHub Copilot. Una mejor solución sería declarar nuestra clase como una data-class, es decir, una clase que solo almacena datos, y no tiene por qué tener un comportamiento específico, y aquí es donde entran los records.
De este modo la clase Person se podría reescribir como un record de la siguiente manera:
public record Person(String name, String lastName) { }
Esto automáticamente genera los métodos equals, hashCode y toString, así como los métodos getter para cada uno de los atributos.
Si lo que se requiere es una estructura de datos inmutables para almacenar datos y no se necesita realizar modificaciones en los atributos (simplemente se ve como un objeto para transportar información). Por otro lado, si se necesita una estructura más general que posea una lógica única y métodos específicos, un enfoque para un paradigma orientado a objetos, aplicar patrones de diseño, o trabajar con JPA o Hibernate, etc., entonces se debería utilizar una clase.
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(Listproducts) { 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.
Descargo de responsabilidad: Todos los recursos proporcionados provienen en parte de Internet. Si existe alguna infracción de sus derechos de autor u otros derechos e intereses, explique los motivos detallados y proporcione pruebas de los derechos de autor o derechos e intereses y luego envíelos al correo electrónico: [email protected]. Lo manejaremos por usted lo antes posible.
Copyright© 2022 湘ICP备2022001581号-3