1

我想知道如何将单例绑定到提供程序中的参数。

IE:

@Singleton
public class RunButtonController{
    @Inject
    public RunButtonController(EventBus eventbus){ ... }

    public JComponent getView(){
        return view;
    }
    ...
}

public class UncaughtExceptionController{
    @Inject
    public UncaughtExceptionController(
        EventBus eventBus, ..., 
        @Named(DefaultExceptionViewKey) JComponent defaultView)
    { ... }
    ...
}

public class GUIController{
    //another consumer of RunButtonController, arguably the "main" consumer. 
    @inject
    public GUIController(RunButtonController runButtonController,
                         UncaughtExceptionController uncaughtExceptionController, ...)
    { ... }
    ...
}

public class Bootstrapper{
    public static void main(String[] args){

        Injector injector = Guice.createInjector(new OptipModule());

        GUIController controller = injector.getInstance(GUIController.class);
    }

    private static class OptipModule extends AbstractModule{

        @Override
        protected void configure() {
            bind(EventBus.class).toInstance(new EventBus("OPTIPGlobalEventBus"));
            bind(JFrame.class).to(GUIView.class);
        }

        @Provides @Named(DefaultExceptionViewKey)
        public JComponent getFrom(RunButtonController runButtonController){
            return runButtonController.getView();
        }
    }
}

在我的构造函数上放置一个断点RunButtonController,我可以看到它始终被实例化两次。我希望它只被实例化一次,我想 defaultExceptionViewProvider == runButtonController 成为true.

我已经相当广泛地使用了 Castle Windsor,但这是我使用过的唯一 IOC 容器,所以我对 guice 很陌生。我在整个地方都看到了访问者行为的残余,guice 的文档清楚地表明,类的定义行为(即,实例一次,使用这个实例,使用这个工厂,......)不会持续超过为其配置的模块。我想说我看到它写了当你使用时@Provides,guice 会为你创建一个子模块,所以大概我需要做的是告诉这个孩子@Provides-生成的模块'嘿,这个类是一个单例我正在解决它,所以就在这里!不要使用你自己的!

我认为我正在以错误的方式处理这个框架。我一直在破坏注释并进行调试,但也许我真正需要做的是花几个小时阅读一个好的教程,不幸的是我找不到。JavaDoc 有示例,并且网页发布了它们,但是它们给您的上下文很少,因此,在阅读了三遍@Assisted 上的文章后我仍然不明白它(也许这就是我应该使用的?)加分对于指向特别详细的博客作者和他页面上的guice条目的人。

沿着这些思路,并且非常离题,我想知道我试图将“嘿,你的默认通知区域是其他人的视图”推送到我的 IOC 容器中会产生什么后果。这可能是领域逻辑吗?我真的不想UnhandledExceptionController知道它的视图是由 a 提供的RunButtonController,同样我不想RunButtonController知道它的视图被用于除被印在视图树上之外的任何东西。

谢谢阅读!

4

1 回答 1

2

如发布的那样,您的代码看起来应该可以工作。也就是说,有一些警告可能导致单例共存。仔细检查每个构造函数调用的堆栈跟踪。

  • 这可能很明显,但是您可以在 Guice 的控制之外创建任意数量的实例,而 Guice 无法知道这些实例的存在。仔细检查您的代码中是否没有RunButtonController手动调用构造函数。

  • 在任何给定的注入器中都会强制执行单例行为。如果您的应用程序中有两个或更多注入器,它们可以各自创建自己的RunButtonController. 但是,在父注入器中声明的单例对任何子注入器都是可见的。

  • 单例通过 key工作。如果您要删除@Singleton注释并添加以下内容:

    bind(RunButtonController.class)
        .annotatedWith(Names.named("Foo"))
        .asEagerSingleton()
    

    然后注入RunButtonController每次都会给你一个新实例,但注入@Named("Foo") RunButtonController会给你一个单独的单例实例,每次都返回相同的实例。这可能不适用于您,因为@Singleton它在课程本身上,但它以前曾咬过其他人。

  • 您似乎不依赖继承,但请记住单例注释不会从超类继承到子类

旁注:@Provides方法不能通过子注入器工作,但私有模块可以(在文档中被称为“父注入器” )。在内部,确实有一个单独的内部模块负责调用这些提供程序方法,但这并不重要——模块之间共享单例。

@Named(DefaultExceptionViewKey) JComponent关于您关于共享视图的题外话:通过注入而不是 ,您已经做得很好了RunButtonController,但是如果您想更加与实现无关,您可以创建一个ExceptionHandlerComponent接口和代码。

于 2013-08-14T21:46:20.410 回答