本文中使用了java8的@FunctionalInterface,对这个不熟悉的同学可以参考我的另一篇文章:java8-Function-how-to-use

转载:原文链接

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

没有,但我可以 new 一个!

1
2
3
4
5
6
7
8
9
10
public class GirlFriend {
private String name;
private int age;
// 省略 getter & setter ...
public static void main(String[] args) {
GirlFriend myGirlFriend = new GirlFriend();
myGirlFriend.setName("小美");
myGirlFriend.setAge(18);
}
}

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

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
public class GirlFriend {

private String name;
private int age;
private int bust;
private int waist;
private int 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 ...

public static void main(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("上海浦东");
myGirlFriend.setMobile("18688888888");
myGirlFriend.setEmail("pretty-xiaomei@qq.com");
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 是很美,但写起来也太麻烦了吧。

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

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

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

先看看使用姿势:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
public class GirlFriend {

// 省略属性 ...

// 省略 getter & setter ...

// 为了演示方便,加几个聚合方法
public void addHobby(String hobby) {
this.hobby = Optional.ofNullable(this.hobby).orElse(new ArrayList<>());
this.hobby.add(hobby);
}

public void addGift(String day, String gift) {
this.gift = Optional.ofNullable(this.gift).orElse(new HashMap<>());
this.gift.put(day, gift);
}

public void setVitalStatistics(int bust, int waist, int hips) {
this.bust = bust;
this.waist = waist;
this.hips = hips;
}

public static void main(String[] args) {
GirlFriend myGirlFriend = Builder.of(GirlFriend::new)
.with(GirlFriend::setName, "小美")
.with(GirlFriend::setAge, 18)
.with(GirlFriend::setVitalStatistics, 33, 23, 33)
.with(GirlFriend::setBirthday, "2001-10-26")
.with(GirlFriend::setAddress, "上海浦东")
.with(GirlFriend::setMobile, "18688888888")
.with(GirlFriend::setEmail, "pretty-xiaomei@qq.com")
.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:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
/**
* 通用的 Builder 模式构建器
*
* @author: CipherCui
* @since 2019/8/29
*/
public class Builder<T> {

private final Supplier<T> instantiator;
private List<Consumer<T>> modifiers = new ArrayList<>();

public Builder(Supplier<T> instantiator) {
this.instantiator = instantiator;
}

public static <T> Builder<T> of(Supplier<T> instantiator) {
return new Builder<>(instantiator);
}

public <P1> Builder<T> with(Consumer1<T, P1> consumer, P1 p1) {
Consumer<T> c = instance -> consumer.accept(instance, p1);
modifiers.add(c);
return this;
}

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);
return this;
}

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);
return this;
}

public T build() {
T value = instantiator.get();
modifiers.forEach(modifier -> modifier.accept(value));
modifiers.clear();
return value;
}

/**
* 1 参数 Consumer
*/
@FunctionalInterface
public interface Consumer1<T, P1> {
void accept(T t, P1 p1);
}

/**
* 2 参数 Consumer
*/
@FunctionalInterface
public interface Consumer2<T, P1, P2> {
void accept(T t, P1 p1, P2 p2);
}

/**
* 3 参数 Consumer
*/
@FunctionalInterface
public interface Consumer3<T, P1, P2, P3> {
void accept(T t, P1 p1, P2 p2, P3 p3);
}
}

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


以上内容为转载,下面加上我自己的理解:

设值时Lambda can be replaced with method reference,可以进行如下简化

1
2
3
4
5
6
7
8
9
10
GirlFriend girlFriend = Builder.of(GirlFriend::new)
.with(GirlFriend::setName, "1234")
.with(GirlFriend::setAge, 10)
.build();
// 以上代码可以替换为:
GirlFriend test2 = Builder.of(GirlFriend::new)
.with((girlInfo, o) -> {
girlInfo.setName(o); // 这句可以替换为 GirlFriend::setName
}, "1234")
.build();

过去实现接口时可以通过以下两种方式:

  • 通过匿名内部类的方式实现接口
  • 通过class类implement接口的方式

在java8中可以使用lambda方式来实现接口:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
/**
* 1 参数 Consumer
*/
@FunctionalInterface
public interface Consumer1<T, P1> {
void accept(T t, P1 p1);
}

/**
* @since 1.8
*/
@FunctionalInterface
public interface Consumer<T> {

/**
* Performs this operation on the given argument.
*
* @param t the input argument
*/
void accept(T t);
...
}

public <P1> Builder<T> with(Consumer1<T, P1> consumer, P1 p1) {
// 其中Consumer为java8中的@FunctionalInterface 接口
Consumer<T> c = new Consumer<T>() {
@Override
public void accept(T t) {
consumer.accept(t , p1);
}
};
// 以上代码可以替换为如下代码
//Consumer<T> c = instance -> consumer.accept(instance, p1);
modifiers.add(c);
return this;
}

调用的过程:

这里使用非lambda表达式的方式,方便查看调用过程:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
// 1. 首先从main方法的build() 方法触发build
// 2. 在build()方法中遍历modifiers,通过modifier.accept(girlFriend)调用了 with方法中的accept
// 3. with方法中再去触发consumer1的accept(girlFriend, '1234')
// 最后会调用girlInfo.setName(o)完成对象的赋值
public static void main(String[] args){
GirlFriend test2 = Builder.of(GirlFriend::new)
.with((girlInfo, o) -> {
girlInfo.setName(o); //这句可以替换为 GirlFriend::setName
}, "1234")
//这里将Consumer1的实现 和 “1234” 传入with方法
.build();
}

// p1 为’1234‘
public <P1> Builder<T> with(Consumer1<T, P1> consumer, P1 p1) {
Consumer<T> c = new Consumer<T>() {
//with方法中的accept, t为new出来的GirlFriend对象
@Override
public void accept(T t) {
consumer.accept(t , p1);
}
};
modifiers.add(c);
return this;
}

public T build() {
T value = instantiator.get();
modifiers.forEach(modifier -> {
//此处将new出来的GirlFriend传入with方法中的accept
modifier.accept(value);
});
modifiers.clear();
return value;
}

赶紧用起来把!