我有一个带有List属性的Java AutoValue类。我想允许构建器附加到List而不必传递整个构造列表。
示例:
import com.google.auto.value.AutoValue;
@AutoValue
public abstract class Deck {
public abstract List<Card> cards();
public static Builder builder() {
return new AutoValue_Card.Builder()
.cards(new ArrayList<Card>());
}
@AutoValue.Builder
public abstract static class Builder {
public abstract Builder cards(List<Card> cards);
/**
* Append card to cards in the constructed Deck.
*/
public Builder addCard(Card card) {
// Is it possible to write this function?
}
}
}
编写addCard函数的最佳解决方案是什么? AutoValue是否已经以某种方式支持此功能?构造类中的中间卡属性对Builder不可见,因此我无法直接访问它。我可以尝试通过在Builder中保留我自己的卡片副本来直接绕过Builder,这是唯一的选择吗?
答案 0 :(得分:7)
当您使用Guava的immutable collections时,AutoValue会有一些特殊功能,因为这些集合都有自己的关联构建器类型。 (当然,你当然希望你的值类的属性是不可变的!)
一个功能是您可以定义一个抽象的cardsBuilder()
方法,该方法将启动相应集合类型的新构建器并将其返回给调用者。这是非常灵活的,但令人遗憾的是它“打破了链条”。来电者现在正在ImmutableList.Builder
拨打电话,无法回到Deck.Builder
拨打build
。
但是您可以通过添加自己的addCard
方法来解决这个问题,就像您在上面显示的那样,只需将其实现为调用cardsBuilder().add(card)
,然后返回this
而不是建设者。问题解决了!
我相信这一切应该在1.1中有效,但如果您遇到麻烦,请告诉我们!
P.S。我道歉 - 我们为AutoValue用户指南提供了很多新材料,包括这个主题,由于一些技术上的困难,这些材料一直被推到公共网站。
答案 1 :(得分:4)
如果有人发现它有用 - 这是我认为凯文建议的实际代码。当我第一次偶然发现他的回复时,这一点并不明显,所以你走了:
import com.google.auto.value.AutoValue;
import com.google.common.collect.ImmutableList;
@AutoValue
public abstract class Hand {
public static Builder builder() {
return new AutoValue_Hand.Builder();
}
public abstract Builder toBuilder();
public abstract String name();
public abstract ImmutableList<String> cards();
@AutoValue.Builder
public static abstract class Builder {
public abstract Builder name(String name);
protected abstract ImmutableList.Builder<String> cardsBuilder();
public Builder addCard(String card) {
cardsBuilder().add(card);
return this;
}
public abstract Hand build();
}
}
用法:
Hand fullHouseHand = Hand.builder()
.name("Full House")
.addCard("King")
.addCard("King")
.addCard("King")
.addCard("10")
.addCard("10")
.build();
System.out.print(fullHouseHand);
输出:
Hand{name=Full House, cards=[King, King, King, 10, 10]}