
本教程旨在解决使用lombok `@superbuilder` 构建继承体系时,子类对象在打印时未能显示父类属性的常见误解。我们将深入探讨`@superbuilder`如何实现跨继承链的构建器模式,并重点说明通过在子类上添加`@tostring(callsuper=true)`注解,来确保`tostring()`方法能够正确包含并显示所有继承自父类的属性,从而提供完整的对象表示。
Lombok 的 @SuperBuilder 注解是 @Builder 的一个增强版本,专门用于处理类继承体系中的构建器模式。当一个类继承自另一个也使用了 @SuperBuilder 的类时,子类的构建器将能够访问并设置父类的所有属性。这使得在创建子类实例时,可以统一地通过一个构建器链来设置所有层次的属性,极大地简化了代码。
然而,开发者在使用 @SuperBuilder 构建对象后,可能会遇到一个看似“父类属性未被继承”的问题,尤其是在打印子类对象时。这并非属性未被继承,而是默认的 toString() 方法行为所致。
考虑以下类结构:
import lombok.Data;
import lombok.EqualsAndHashCode;
import lombok.experimental.SuperBuilder;
// 父类
@Data
@SuperBuilder
public class CParent {
protected Integer parentId;
}
// 子类
@Data
@SuperBuilder
@EqualsAndHashCode(callSuper = true)
public class CChild extends CParent {
protected String childId;
}
// 包含 CParent 类型的组合类
@Data
@SuperBuilder
public class CHouse {
String address;
String description;
CParent parent; // 可以是 CParent 或其子类 CChild
}按照上述定义,我们尝试构建一个 CChild 对象并将其赋值给 CHouse:
public class Main {
public static void main(String[] args) {
// 构建 CChild 实例
CChild child = CChild.builder()
.parentId(123) // 设置父类属性
.childId("789") // 设置子类属性
.build();
// 构建 CHouse 实例,并传入 CChild 对象
CHouse house = CHouse.builder()
.address("address")
.description("description")
.parent(child)
.build();
System.out.println("CChild object: " + child);
System.out.println("CHouse object: " + house);
}
}在上述代码中,当我们运行 Main 方法并打印 child 对象时,预期输出可能是 CChild(childId=789, parentId=123)。然而,实际输出可能仅显示 CChild(childId=789),导致开发者误认为 parentId 未被正确设置或继承。
出现上述现象的原因在于 Lombok 的 @Data 注解。@Data 是一个复合注解,它包含了 @ToString、@EqualsAndHashCode、@Getter、@Setter 和 @RequiredArgsConstructor。当 @Data 应用于一个类时,Lombok 会自动生成这些方法的实现。
对于 toString() 方法,默认生成的实现不会自动包含父类的属性。这意味着,即使 CChild 实例内部确实拥有 parentId 的值,其默认的 toString() 方法也只会打印 CChild 类自身定义的属性(即 childId)。因此,parentId 实际上已经被正确继承和设置,只是在对象被打印时没有被显示出来。
要解决这个问题,确保 toString() 方法能够完整地显示所有继承自父类的属性,我们需要在子类上明确指定 @ToString(callSuper=true)。这个参数会指示 Lombok 在生成 toString() 方法时,首先调用父类的 toString() 方法,然后将父类的输出与子类的输出合并。
修正后的类定义如下:
import lombok.Data;
import lombok.EqualsAndHashCode;
import lombok.ToString; // 引入 ToString
import lombok.experimental.SuperBuilder;
// 父类
@Data
@SuperBuilder
public class CParent {
protected Integer parentId;
}
// 子类 - 关键修改:添加 @ToString(callSuper = true)
@Data
@SuperBuilder
@EqualsAndHashCode(callSuper = true) // 确保 equals 和 hashCode 也包含父类属性
@ToString(callSuper = true) // 确保 toString 包含父类属性
public class CChild extends CParent {
protected String childId;
}
// 包含 CParent 类型的组合类
@Data
@SuperBuilder
public class CHouse {
String address;
String description;
CParent parent;
}现在,当我们再次运行 Main 方法:
public class Main {
public static void main(String[] args) {
CChild child = CChild.builder()
.parentId(123)
.childId("789")
.build();
CHouse house = CHouse.builder()
.address("address")
.description("description")
.parent(child)
.build();
System.out.println("CChild object: " + child);
System.out.println("CHouse object: " + house);
}
}输出将符合预期:
CChild object: CChild(super=CParent(parentId=123), childId=789) CHouse object: CHouse(address=address, description=description, parent=CChild(super=CParent(parentId=123), childId=789))
这明确表明 parentId 属性已被 CChild 实例正确继承和赋值,并且在打印时也得到了完整的显示。
Lombok 的 @SuperBuilder 是一个强大的工具,能够优雅地处理 Java 中的构建器模式与继承。当遇到子类对象打印时父类属性“丢失”的情况,这通常不是属性未被继承,而是 toString() 方法的默认行为所致。通过在子类上显式添加 @ToString(callSuper=true),我们可以确保 toString() 方法能够递归地包含所有父类的属性,从而提供对象状态的完整且准确的字符串表示。理解并正确应用这些 Lombok 注解的 callSuper 参数,对于构建健壮且易于调试的 Java 应用程序至关重要。
以上就是Lombok @SuperBuilder 在继承结构中正确构建与显示父类属性的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号