【问题标题】:Guice, Injecting TypeLiteral<T> when using @AssistedInjectGuice,使用 @AssistedInject 时注入 TypeLiteral<T>
【发布时间】:2012-04-18 22:51:35
【问题描述】:

这行得通:

public static class SomeGenericType<T> {
    private TypeLiteral<T> type; 

    @Inject
    public SomeGenericType(TypeLiteral<T> type) {
        this.type = type; 
    }

    public Class<? super T> getType() {
        return type.getRawType();
    }
}

当我这样做时,Guice 会自动注入表示字符串的 TypeLiteral:

@Inject SomeGenericType<String> foo;

但是当使用辅助注入尝试同样的事情时:

public static interface FooFactory<T> {
    Foo<T> create(String name);
}

public static class Foo<T> {

    @AssistedInject
    public Foo(TypeLiteral<T> type, @Assisted String name) {
        ....

我的模块如下所示:

public static class TestMod extends AbstractModule {
    @Override
    protected void configure() {
        install(new FactoryModuleBuilder().build(new TypeLiteral<FooFactory<String>>(){}));
    }   
}

安装模块时出现异常:

TypeLiteral<T> cannot be used as a Key, it is not fully specified. 

问题肯定是我尝试注入的 TypeLiteral,因为当我删除它时,通用工厂确实可以正常工作。

所以,我现在可能只是建立自己的工厂,但我很好奇这是否可行?使用 FactoryModuleBuilder 是否略有不同?

【问题讨论】:

    标签: guice guice-3


    【解决方案1】:

    您如何访问 FooFactory 的实例?我在下面的代码上构建了变体,它对我有用:

    public class AnotherGuiceTest {
        public static void main( String[] args ) {
            Injector i = Guice.createInjector( new TestMod() );
            FooFactory<String> ff = i.getInstance( Key.get( new TypeLiteral<FooFactory<String>>() {} ) );
            ff.create( "myname" );
        }
    }
    
    interface FooFactory<T> {
        Foo<T> create( String name );
    }
    
    class Foo<T> {
    
        @Inject
        public Foo( TypeLiteral<T> type, @Assisted String name ) {
            System.out.println( type.getRawType() );
            System.out.println( name );
        }
    }
    
    class TestMod extends AbstractModule {
        @Override
        protected void configure() {
            install( new FactoryModuleBuilder().build( new TypeLiteral<FooFactory<String>>() {} ) );
        }
    }
    

    输出:

    class java.lang.String
    myname
    

    注意我使用了常规的@Inject 注释而不是@AssistedInject,我认为 用于工厂中的多个构造函数。如果您直接注入实例,这也有效:

    public class AnotherGuiceTest {
        public static void main( String[] args ) {
            Injector i = Guice.createInjector( new TestMod() );
            AppClass ac = i.getInstance( AppClass.class );
        }
    }
    
    class AppClass {
        @Inject
        public AppClass( FooFactory<String> fooFactory ) {
            fooFactory.create( "test" );
        }
    }
    

    输出:

    class java.lang.String
    test
    

    【讨论】:

    • 啊哈..所以这是一个设置问题。感谢您花时间测试和回答。原来这是因为我有一个来自 GIN 的罐子,guice-assistedinject-snapshot.jar 而不是 guice-assistedinject-3.0.jar (旁注,常规辅助注入似乎确实适用于这个罐子,只是不适用于泛型)
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2017-09-02
    • 2016-08-09
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多