Google Guice-如何自动添加绑定

时间:2018-10-21 13:05:56

标签: java guice

在我的项目中,我使用Google Guice进行依赖项注入。在扩展Google Guice的AbstactModule的Module类中,我有一个MapBinder,其中的键是项目的名称,值是我的Item接口的实现(例如FirstItem)。

MapBinder<String, Item> itemMapBinder = MapBinder.newMapBinder(binder(), String.class, Item.class);
itemMapBinder.addBinding("FirstItem").to(FirstItem.class);

当前,每次我添加Item的新实现(例如SecondItem)时,我还必须向Module添加新行,例如

itemMapBinder.addBinding("SecondItem").to(SecondItem.class);

我在我的ItemFactory中使用了这张物品地图。

public class ItemFactoryImpl implements ItemFactory {

    private final Map<String, Item> items;

    @Inject
    public ItemFactoryImpl(Map<String, Item> items) {
        this.items = items;
    }

    @Override
    public Item getItem(String name) {
        if (name == null) throw new IllegalArgumentException("Name cannot be null");
        return items.get(name);
    }
}

我试图找到一种方法,该方法如何自动为Item接口的新实现添加绑定。我的想法是使用自定义注释,该注释将被添加到Item接口的新实现中,并以某种方式修改Module#configure来添加具有该自定义注释的所有类的绑定。我完成了这样的事情:

// cz.milanhlinak.guiceautobinding.item.AutoBindableItem.java
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface AutoBindableItem {
}

// cz.milanhlinak.guiceautobinding.item.Item.java
public interface Item {
}

// cz.milanhlinak.guiceautobinding.item.SecondItem.java
@AutoBindableItem
public class SecondItem implements Item {
}

// cz.milanhlinak.guiceautobinding.Module.java
public class Module extends AbstractModule {

    @Override
    protected void configure() {

        MapBinder<String, Item> itemMapBinder = MapBinder.newMapBinder(binder(), String.class, Item.class);
        new Reflections("cz.milanhlinak.guiceautobinding.item")
            .getTypesAnnotatedWith(AutoBindableItem.class)
            .stream()
            .filter(Item.class::isAssignableFrom)
            .forEach(typeAnnotatedWith -> itemMapBinder
                    .addBinding(typeAnnotatedWith.getSimpleName())
                    .to((Class<? extends Item>) typeAnnotatedWith)
            );

        bind(ItemFactory.class).to(ItemFactoryImpl.class).in(Singleton.class);
    }
}

我的GitHub repository中提供了完整的代码。

我想知道是否有更好的方法来实现与Google Guice的自动绑定,因为如您所见,我当前正在使用一个附加库-Reflections

更新

另一个选择可能是使用Google Guava而不是Reflections

public class Module extends AbstractModule {

    @Override
    protected void configure() {

        MapBinder<String, Item> itemMapBinder = MapBinder.newMapBinder(binder(), String.class, Item.class);

        ClassPath classPath;
        try {
            classPath = ClassPath.from(Thread.currentThread().getContextClassLoader());
        } catch (IOException e) {
            throw new RuntimeException("Unable to read class path resources", e);
        }

        ImmutableSet<ClassPath.ClassInfo> topLevelClasses = classPath.getTopLevelClassesRecursive("cz.milanhlinak.guiceautobinding.item");
        topLevelClasses.stream()
                .map(ClassPath.ClassInfo::load)
                .filter(clazz -> clazz.isAnnotationPresent(AutoBindableItem.class) && Item.class.isAssignableFrom(clazz))
                .forEach(clazz -> itemMapBinder
                        .addBinding(clazz.getSimpleName())
                        .to((Class<? extends Item>) clazz));

        bind(ItemFactory.class).to(ItemFactoryImpl.class).in(Singleton.class);
    }
}

1 个答案:

答案 0 :(得分:1)

您没有做错任何事。

使用Reflections没有错。该工具非常出色,可以完成您想要的工作。

Guice没有发现工具,因此没有“默认”方式只能在Guice中执行您想要的操作。

如果您要使用反射以外的其他功能,则可以退回到- if can? :menu, Trade || Order # So I added this as well. li.dropdown class = check_active('withdraws') = link_to 'javascript:;', class: 'dropdown-toggle', 'data-toggle' => 'dropdown' do span = t('admin_header.trades') span.caret ul.dropdown-menu - if can? :manage, Order li class = check_active('orders') = link_to t(".menus.items.operating.orders"), admin_statistic_orders_path ,但是java.util.ServiceLoader并不是Guice友好的,因为它会自动创建实例,但是您通常使用Guice是因为您想要它可以为您创建实例,而不是让另一个框架为您做到这一点。

使用Guava的ServiceLoader的替代方法也很有意义,但是Reflections功能更强大,因为您还可以加载不在类路径中的类。

最重要的是,您已经做出了更好的选择。