1 回答

TA貢獻1827條經驗 獲得超9個贊
如果沒有一些昂貴的類路徑掃描,這將是不可能的:如果注入器沒有對您的 Twix 類的任何引用,它就無法在不掃描類路徑上的每個 JAR 的情況下將其綁定到 Map 中@SnackImpl-帶注釋的類。您可以嘗試使用Guava 的 ClassPath,但如果您使用基于網絡或自定義的類加載器,這可能根本無法處理。無論如何,我不會推薦它。
一種替代方法是使用 Java 的內置ServiceLoader框架,該框架允許各個 JAR 列出給定服務(接口)的完全合格的實現。您甚至可以使用 Google 的 Auto 框架根據注釋為您生成該服務文件。
這需要列出實現,但您仍然需要將它們綁定到 MapBinder。幸運的是,MapBinder 不需要單個定義,并且會在模塊構建期間自動合并多個 MapBinder 定義:
支持從不同模塊貢獻映射綁定。例如,可以讓 CandyModule 和 ChipsModule 都創建自己的 MapBinder,并各自為小吃地圖提供綁定。注入該映射后,它將包含來自兩個模塊的條目。
(來自MapBinder 文檔)
考慮到這一點,我建議每個插件包都有自己的 Guice 模塊,并在其中注冊到 MapBinder,然后使用 ServiceLoader 將這些 Guice 模塊添加到主注入器,以便在注入器創建時獲取這些模塊。
// Assume CandyPluginModule extends AbstractModule
@AutoService(CandyPluginModule.class)
public TwixPluginModule extends CandyPluginModule {
@Override public void configure() {
MapBinder<String, Snack> mapBinder
= MapBinder.newMapBinder(binder(), String.class, Snack.class);
mapBinder.addBinding("twix").to(Twix.class);
}
}
您還可以利用超類:
@AutoService(CandyPluginModule.class)
public TwixPluginModule extends CandyPluginModule {
@Override public void configureSnacks() { // defined on CandyPluginModule
bindSnack("twix").to(Twix.class);
}
}
或者,您可以直接使用 AutoService 列出 Twix 之類的實現,然后創建一個模塊,將所有 ServiceLoader 實現讀入您的 MapBinder,但這可能會限制您的插件的靈活性,并且不會讓您獲得 MapBinder 沒有的綁定的任何分散化還沒給你。
添加回答
舉報