阿里云-云小站(无限量代金券发放中)
【腾讯云】云服务器、云数据库、COS、CDN、短信等热卖云产品特惠抢购

记录类

32次阅读
没有评论

共计 1967 个字符,预计需要花费 5 分钟才能阅读完成。

使用 StringInteger 等类型的时候,这些类型都是不变类,一个不变类具有以下特点:

  1. 定义 class 时使用final,无法派生子类;
  2. 每个字段使用final,保证创建实例后无法修改任何字段。

假设我们希望定义一个 Point 类,有 xy 两个变量,同时它是一个不变类,可以这么写:

public final class Point {private final int x;
    private final int y;

    public Point(int x, int y) {this.x = x;
        this.y = y;
    }

    public int x() {return this.x;
    }

    public int y() {return this.y;
    }
}

为了保证不变类的比较,还需要正确覆写 equals()hashCode()方法,这样才能在集合类中正常使用。后续我们会详细讲解正确覆写 equals()hashCode(),这里演示 Point 不变类的写法目的是,这些代码写起来都非常简单,但是很繁琐。

record

从 Java 14 开始,引入了新的 Record 类。我们定义 Record 类时,使用关键字 record。把上述Point 类改写为 Record 类,代码如下:

// Record
public class Main {public static void main(String[] args) {Point p = new Point(123, 456);
        System.out.println(p.x());
        System.out.println(p.y());
        System.out.println(p);
    }
}

record Point(int x, int y) {}

仔细观察 Point 的定义:

record Point(int x, int y) {}

把上述定义改写为 class,相当于以下代码:

final class Point extends Record {private final int x;
    private final int y;

    public Point(int x, int y) {this.x = x;
        this.y = y;
    }

    public int x() {return this.x;
    }

    public int y() {return this.y;
    }

    public String toString() {return String.format("Point[x=%s, y=%s]", x, y);
    }

    public boolean equals(Object o) {...}
    public int hashCode() {...}
}

除了用 final 修饰 class 以及每个字段外,编译器还自动为我们创建了构造方法,和字段名同名的方法,以及覆写 toString()equals()hashCode()方法。

换句话说,使用 record 关键字,可以一行写出一个不变类。

enum 类似,我们自己不能直接从 Record 派生,只能通过 record 关键字由编译器实现继承。

构造方法

编译器默认按照 record 声明的变量顺序自动创建一个构造方法,并在方法内给字段赋值。那么问题来了,如果我们要检查参数,应该怎么办?

假设 Point 类的 xy 不允许负数,我们就得给 Point 的构造方法加上检查逻辑:

public record Point(int x, int y) {public Point {if (x < 0 || y < 0) {throw new IllegalArgumentException();}
    }
}

注意到方法 public Point {...} 被称为 Compact Constructor,它的目的是让我们编写检查逻辑,编译器最终生成的构造方法如下:

public final class Point extends Record {public Point(int x, int y) {// 这是我们编写的 Compact Constructor:
        if (x < 0 || y < 0) {throw new IllegalArgumentException();}
        // 这是编译器继续生成的赋值代码:
        this.x = x;
        this.y = y;
    }
    ...
}

作为 recordPoint仍然可以添加静态方法。一种常用的静态方法是 of() 方法,用来创建Point

public record Point(int x, int y) {public static Point of() {return new Point(0, 0);
    }
    public static Point of(int x, int y) {return new Point(x, y);
    }
}

这样我们可以写出更简洁的代码:

var z = Point.of();
var p = Point.of(123, 456);

小结

从 Java 14 开始,提供新的 record 关键字,可以非常方便地定义 Data Class:

  • 使用 record 定义的是不变类;
  • 可以编写 Compact Constructor 对参数进行验证;
  • 可以定义静态方法。

正文完
星哥说事-微信公众号
post-qrcode
 0
星锅
版权声明:本站原创文章,由 星锅 于2024-08-05发表,共计1967字。
转载说明:除特殊说明外本站文章皆由CC-4.0协议发布,转载请注明出处。
【腾讯云】推广者专属福利,新客户无门槛领取总价值高达2860元代金券,每种代金券限量500张,先到先得。
阿里云-最新活动爆款每日限量供应
评论(没有评论)
验证码
【腾讯云】云服务器、云数据库、COS、CDN、短信等云产品特惠热卖中