是否可以从静态方法开始使用 Lombok @Builder?
Is it possible to use Lombok @Builder, starting from static method?
我希望 Lombok 照顾我的构建器 API,同时也有一个构造器作为开始。
我从 @Data
class 和 @Builder(toBuilder = true)
上的构造函数开始,但这让我不得不在最终字段上强制使用无效或虚拟值以及表现力较差的流利 API。我终于使用静态方法解决了我的情况,但我希望 Lombok 对我的用例有更好的解决方案。
API 使用 toBuilder
fooHandler.accept(new TweakedFoo(Foo.class, Mode.QUICK).toBuilder()
.mappingOutcomeFor(FooOutcome.class)
.mappingOutcome(toEvent(BarOutcome.class))
.build()));
API 使用静态方法
fooHandler.accept(tweakFoo(Foo.class, Mode.QUICK)
.mappingOutcomeFor(FooOutcome.class)
.mappingOutcome(toEvent(BarOutcome.class))
.build()));
看看第二个设置如何更好地进行?
各自的 Lombok 设置(简化)
@Data
@Builder(toBuilder = true)
public class TweakedFoo {
private final Class<Foo> from;
private final Mode mode;
private final Class<?> to;
public TweakedFoo(Class<Foo> from, Mode mode) {
this.from = from;
this.mode = mode;
this.to = null; // eww
}
}
和:
@Data
@Builder(builderMethodName = "useTweakedFooDotTweakedFooInsteadPlease")
@AllArgsConstructor(access = AccessLevel.PRIVATE)
public class TweakedFoo {
private final Class<Foo> from;
private final Mode mode;
private final Class<?> to;
public static TweakedFooBuilder tweakFoo(Class<Foo> from, Mode mode) {
return TweakedFoo.useTweakedFooDotTweakedFooInsteadPlease()
.from(from)
.mode(mode);
}
}
实际参数在这里没有多大意义,但此设置说明了我的真实用例。
第二种方法不仅更简洁,它不需要虚拟构造函数字段初始化,而且它隐藏了构造函数,所以除了通过构建器之外你无法获得实例。但是,第二种方法需要我隐藏 Lombok 生成的构建器启动方法以支持我自己的静态方法。
Lombok 有更好的方法吗?
您可以自定义 builder()
方法,只需自己实现即可:
@Data
@Builder
public class TweakedFoo {
// ...
public static TweakedFooBuilder builder(Class<Foo> from, Mode mode) {
return new TweakedFooBuilder()
.from(from)
.mode(mode);
}
// ...
}
在这种情况下,Lombok 将不会生成另一个 builder()
方法,因为它识别具有相同名称的现有方法。如果您希望方法以不同的方式命名,例如tweakFoo
, 使用 @Builder(builderMethodName="tweakFoo")
.
我希望 Lombok 照顾我的构建器 API,同时也有一个构造器作为开始。
我从 @Data
class 和 @Builder(toBuilder = true)
上的构造函数开始,但这让我不得不在最终字段上强制使用无效或虚拟值以及表现力较差的流利 API。我终于使用静态方法解决了我的情况,但我希望 Lombok 对我的用例有更好的解决方案。
API 使用 toBuilder
fooHandler.accept(new TweakedFoo(Foo.class, Mode.QUICK).toBuilder()
.mappingOutcomeFor(FooOutcome.class)
.mappingOutcome(toEvent(BarOutcome.class))
.build()));
API 使用静态方法
fooHandler.accept(tweakFoo(Foo.class, Mode.QUICK)
.mappingOutcomeFor(FooOutcome.class)
.mappingOutcome(toEvent(BarOutcome.class))
.build()));
看看第二个设置如何更好地进行?
各自的 Lombok 设置(简化)
@Data
@Builder(toBuilder = true)
public class TweakedFoo {
private final Class<Foo> from;
private final Mode mode;
private final Class<?> to;
public TweakedFoo(Class<Foo> from, Mode mode) {
this.from = from;
this.mode = mode;
this.to = null; // eww
}
}
和:
@Data
@Builder(builderMethodName = "useTweakedFooDotTweakedFooInsteadPlease")
@AllArgsConstructor(access = AccessLevel.PRIVATE)
public class TweakedFoo {
private final Class<Foo> from;
private final Mode mode;
private final Class<?> to;
public static TweakedFooBuilder tweakFoo(Class<Foo> from, Mode mode) {
return TweakedFoo.useTweakedFooDotTweakedFooInsteadPlease()
.from(from)
.mode(mode);
}
}
实际参数在这里没有多大意义,但此设置说明了我的真实用例。
第二种方法不仅更简洁,它不需要虚拟构造函数字段初始化,而且它隐藏了构造函数,所以除了通过构建器之外你无法获得实例。但是,第二种方法需要我隐藏 Lombok 生成的构建器启动方法以支持我自己的静态方法。
Lombok 有更好的方法吗?
您可以自定义 builder()
方法,只需自己实现即可:
@Data
@Builder
public class TweakedFoo {
// ...
public static TweakedFooBuilder builder(Class<Foo> from, Mode mode) {
return new TweakedFooBuilder()
.from(from)
.mode(mode);
}
// ...
}
在这种情况下,Lombok 将不会生成另一个 builder()
方法,因为它识别具有相同名称的现有方法。如果您希望方法以不同的方式命名,例如tweakFoo
, 使用 @Builder(builderMethodName="tweakFoo")
.