Java 中的记录与类
如果您已经了解 java 中的记录,您可能会发现它的用法与类非常相似,但必须考虑到一些重要的差异。在本文中,我们将了解 java 中记录和类之间的差异。如果您仍然不知道记录,我建议阅读我的文章《java 中的记录:它们是什么以及如何使用它们》。
不变性
不可变对象是指一旦创建对象,其属性就无法修改的对象。对于 records 来说,它们是不可变的,也就是说,一旦创建 record 类型的对象,其属性就无法修改。另一方面,类可能是不变的,也可能不是不变的,具体取决于它的实现方式。这部分确保数据的完整性并防止其被意外修改。
目的
类通常只是为了存储数据而编写,例如来自数据库查询的数据或来自表单的数据。在许多情况下,该数据是不可变的,因为需要在不使用同步的情况下确保数据的有效性。为了实现这一点,使用以下元素编写一个类:
- 每个字段的私有属性。
- 每个字段的获取器。
- 初始化所有字段的构造函数。
- 比较对象是否相等的 equals 方法。
- hashcode 方法,根据字段生成哈希码。
- 一个 tostring 方法,用于生成字段的字符串表示形式。
例如,如果你有一个 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 方法。
记录和类之间有什么区别?
- 不可变性:记录是不可变的,也就是说,记录类型的对象一旦创建,其属性就无法修改。相反,类可能是不变的,也可能不是不变的,具体取决于它的实现方式。
- 生成方法:记录自动生成equals、hashcode和tostring方法,以及每个属性的getter方法。另一方面,在类中,这些方法必须手动实现或借助 ide 实现。
- 在oop中使用:记录不能从其他类继承,也不能被其他类扩展,但它们可以实现接口。另一方面,类可以从其他类继承、扩展,并且通常非常适合涵盖面向对象编程的概念。
- 语法:记录的语法比类更简单,因为它可以在一行中定义,而类需要多行代码。
- 用途:记录是一种类似于dto(数据传输对象)的结构,即帮助建模不可变数据的类,其一部分是类 是一个更通用的结构,可以有行为和状态。
何时使用记录,何时使用类?
如果需要的是一个不可变的数据结构来存储数据,并且不需要对属性进行任何修改(简单地看成是一个承载信息的对象)。另一方面,如果您需要具有独特逻辑和特定方法的更通用的结构、面向对象范例的方法、应用设计模式或使用 jpa 或 hibernate 等,那么您应该使用类。
额外:具有可变属性的记录
让我们考虑以下示例,有两条记录 product,其属性为名称和价格,还有一个 cart,其具有 arraylist 类型的单个属性 products
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(); } } </product>
本例中的问题是,每个 记录 本身都是不可变的,但在记录 cart 的情况下,具有 arraylist 类型的属性,并且由于 arraylist 本质上是可变的,一旦记录购物车实例化,就可以修改列表的内容。
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()); } }
上面的代码编译没有问题,因为只修改了列表的内容,但产品属性本身没有被修改。这只是特定情况的示例,可能没有必要,但很高兴知道这是可以做到的。
以上就是Java 中的记录与类的详细内容,更多请关注其它相关文章!