当前位置: 欣欣网 > 码农

你还在 new 对象吗?Java 8 通用 Builder 了解一下...

2024-04-23码农

程序员经常会遇到灵魂拷问:你有对象吗?

没有,但我可以 new 一个!

public classGirlFriend{
private String name;
privateint age;
// 省略 getter & setter ...
publicstaticvoidmain(String[] args){
GirlFriend myGirlFriend = new GirlFriend();
myGirlFriend.setName("小美");
myGirlFriend.setAge(18);
}
}

没问题,老铁!但如果对象的属性太多,咋办?

public classGirlFriend{
private String name;
privateint age;
privateint bust;
privateint waist;
privateint hips;
private List<String> hobby;
private String birthday;
private String address;
private String mobile;
private String email;
private String hairColor;
private Map<String, String> gift;
// 等等等等 ...
// 省略 getter & setter ...
publicstaticvoidmain(String[] args){
GirlFriend myGirlFriend = new GirlFriend();
myGirlFriend.setName("小美");
myGirlFriend.setAge(18);
myGirlFriend.setBust(33);
myGirlFriend.setWaist(23);
myGirlFriend.setHips(33);
myGirlFriend.setBirthday("2001-10-26");
myGirlFriend.setAddress("公众号:一行Java");
myGirlFriend.setMobile("18688888888");
myGirlFriend.setEmail("[email protected]");
myGirlFriend.setHairColor("浅棕色带点微卷");
List<String> hobby = new ArrayList<>();
hobby.add("逛街");
hobby.add("购物");
hobby.add("买东西");
myGirlFriend.setHobby(hobby);
Map<String, String> gift = new HashMap<>();
gift.put("情人节礼物""LBR 1912女王时代");
gift.put("生日礼物""迪奥烈焰蓝金");
gift.put("纪念日礼物""阿玛尼红管唇釉");
myGirlFriend.setGift(gift);
// 等等等等 ...
}
}
GirlFriend{name='小美'
, age=18
, bust=33
, waist=23
, hips=33
, hobby=[逛街, 购物, 买东西]
, birthday='2001-10-26'
, address='上海浦东'
, mobile='18688888888'
, email='[email protected]'
, hairColor='浅棕色带点微卷'
, gift={情人节礼物=LBR 1912女王时代, 生日礼物=迪奥烈焰蓝金, 纪念日礼物=阿玛尼红管唇釉}
}

GirlFriend 是很美,但写起来也太麻烦了吧。

说说缺点:实例化和设置属性分开,不好维护;变量名重复写。

莫慌,看法宝~

这里不再介绍其他 Builder 实现方式,直接祭出最实用的 通用Builder

适用于所有类,不需要改造原来类,不需要 lombok 插件支持。

先看看使用姿势:

public classGirlFriend{
// 省略属性 ...
// 省略 getter & setter ...
// 为了演示方便,加几个聚合方法
publicvoidaddHobby(String hobby){
this.hobby = Optional.ofNullable(this.hobby).orElse(new ArrayList<>());
this.hobby.add(hobby);
}
publicvoidaddGift(String day, String gift){
this.gift = Optional.ofNullable(this.gift).orElse(new HashMap<>());
this.gift.put(day, gift);
}
publicvoidsetVitalStatistics(int bust, int waist, int hips){
this.bust = bust;
this.waist = waist;
this.hips = hips;
}
publicstaticvoidmain(String[] args){
GirlFriend myGirlFriend = Builder.of(GirlFriend::new)
.with(GirlFriend::setName, "小美")
.with(GirlFriend::setAge, 18)
.with(GirlFriend::setVitalStatistics, 332333)
.with(GirlFriend::setBirthday, "2001-10-26")
.with(GirlFriend::setAddress, "上海浦东")
.with(GirlFriend::setMobile, "18688888888")
.with(GirlFriend::setEmail, "[email protected]")
.with(GirlFriend::setHairColor, "浅棕色带点微卷")
.with(GirlFriend::addHobby, "逛街")
.with(GirlFriend::addHobby, "购物")
.with(GirlFriend::addHobby, "买东西")
.with(GirlFriend::addGift, "情人节礼物""LBR 1912女王时代")
.with(GirlFriend::addGift, "生日礼物""迪奥烈焰蓝金")
.with(GirlFriend::addGift, "纪念日礼物""阿玛尼红管唇釉")
// 等等等等 ...
.build();
}
}

看到了吗!实例化和属性设置在同一条语句执行,链式操作,一路点点点,清爽!

Talk is cheap, show me the code:

/**
 * 通用的 Builder 模式构建器
 *
 */

public classBuilder<T{
privatefinal Supplier<T> instantiator;
private List<Consumer<T>> modifiers = new ArrayList<>();
publicBuilder(Supplier<T> instantiator){
this.instantiator = instantiator;
}
publicstatic <T> Builder<T> of(Supplier<T> instantiator){
returnnew Builder<>(instantiator);
}
public <P1> Builder<T> with(Consumer1<T, P1> consumer, P1 p1){
Consumer<T> c = instance -> consumer.accept(instance, p1);
modifiers.add(c);
returnthis;
}
public <P1, P2> Builder<T> with(Consumer2<T, P1, P2> consumer, P1 p1, P2 p2){
Consumer<T> c = instance -> consumer.accept(instance, p1, p2);
modifiers.add(c);
returnthis;
}
public <P1, P2, P3> Builder<T> with(Consumer3<T, P1, P2, P3> consumer, P1 p1, P2 p2, P3 p3){
Consumer<T> c = instance -> consumer.accept(instance, p1, p2, p3);
modifiers.add(c);
returnthis;
}
public T build(){
T value = instantiator.get();
modifiers.forEach(modifier -> modifier.accept(value));
modifiers.clear();
return value;
}
/**
* 1 参数 Consumer
*/

@FunctionalInterface
publicinterfaceConsumer1<TP1{
voidaccept(T t, P1 p1);
}
/**
* 2 参数 Consumer
*/

@FunctionalInterface
publicinterfaceConsumer2<TP1P2{
voidaccept(T t, P1 p1, P2 p2);
}
/**
* 3 参数 Consumer
*/

@FunctionalInterface
publicinterfaceConsumer3<TP1P2P3{
voidaccept(T t, P1 p1, P2 p2, P3 p3);
}
}

这个示例最多支持三个参数的设置属性方法,也完全够用了。如果要扩展也很容易,依葫芦画瓢,添加多个参数的 Consumer

快用你的 Builder 建个对象吧~

来源:ciphermagic.cn/java8-builder.html

>>

END

精品资料,超赞福利,免费领

微信扫码/长按识别 添加【技术交流群

群内每天分享精品学习资料

最近开发整理了一个用于速刷面试题的小程序;其中收录了上千道常见面试题及答案(包含基础并发JVMMySQLRedisSpringSpringMVCSpringBootSpringCloud消息队列等多个类型),欢迎您的使用。

👇👇

👇点击"阅读原文",获取更多资料(持续更新中