UnsatisfiedDependencyException:在 SystemInjecteeImpl 中没有可用于注入的对象

问题描述 投票:0回答:4

在 Jersey Rest 应用中使用 DI 时出现错误:

org.glassfish.hk2.api.UnsatisfiedDependencyException: There was no object available for injection at SystemInjecteeImpl(requiredType=PricingService,parent=PricingResource,qualifiers={},position=0,optional=false,self=false,unqualified=null,1633188703)

我对这个概念很陌生,它看起来很复杂,因为有些例子似乎已被弃用。据我所知,有几种方法可以使 DI 工作:本地 HK2、Spring/HK2 Bridge。什么配置起来更容易、更直接?如何以编程方式(不喜欢 XML)为 Jersey 2.x 设置?

资源配置

import org.glassfish.jersey.server.ResourceConfig;

public class ApplicationConfig  extends ResourceConfig {
    public ApplicationConfig() {
        register(new ApplicationBinder());
        packages(true, "api");
    }
}

抽象活页夹

public class ApplicationBinder extends AbstractBinder {
    @Override
    protected void configure() {
        bind(PricingService.class).to(PricingService.class).in(Singleton.class);
    }
}

定价资源

@Path("/prices")
public class PricingResource {
    private final PricingService pricingService;

    @Inject
    public PricingResource(PricingService pricingService) {
        this.pricingService = pricingService;
    }

    @GET
    @Produces(MediaType.APPLICATION_JSON)
    public Collection<Price> findPrices() {
        return pricingService.findPrices();
    }
}

定价服务

@Singleton
public class PricingService {
   // no constructors...
// findPrices() ...

}

更新

public class Main {
    public static final String BASE_URI = "http://localhost:8080/api/";

    public static HttpServer startServer() {
        return createHttpServerWith(new ResourceConfig().packages("api").register(JacksonFeature.class));
    }

    private static HttpServer createHttpServerWith(ResourceConfig rc) {
        HttpServer httpServer = GrizzlyHttpServerFactory.createHttpServer(URI.create(BASE_URI), rc);
        StaticHttpHandler staticHttpHandler = new StaticHttpHandler("src/main/webapp");
        staticHttpHandler.setFileCacheEnabled(false);
        staticHttpHandler.start();
        httpServer.getServerConfiguration().addHttpHandler(staticHttpHandler);
        return httpServer;
    }

    public static void main(String[] args) throws IOException {
        System.setProperty("java.util.logging.config.file", "src/main/resources/logging.properties");
        final HttpServer server = startServer();

        System.out.println(String.format("Jersey app started with WADL available at "
                + "%sapplication.wadl\nHit enter to stop it...", BASE_URI));
        server.start();
        System.in.read();
        server.stop();
    }

}

更新3:

public class PricingResourceTest extends JerseyTest {
    @Mock
    private PricingService pricingServiceMock;

    @Override
    protected Application configure() {
        MockitoAnnotations.initMocks(this);
        enable(TestProperties.LOG_TRAFFIC);
        enable(TestProperties.DUMP_ENTITY);

        ResourceConfig config = new ResourceConfig(PricingResource.class);
        config.register(new AbstractBinder() {
            @Override
            protected void configure() {
                bind(pricingServiceMock).to(PricingService.class);
            }
        });
        return config;
    }

    @Test
    public void testFindPrices(){
        when(pricingServiceMock.findPrices()).thenReturn(getMockedPrices());
        Response response  = target("/prices")
                .request()
                .get();
        verify(pricingServiceMock).findPrices();
        List<Price> prices = response.readEntity(new GenericType<List<Price>>(){});
//        assertEquals("Should return status 200", 200, response.getStatus());
        assertTrue(prices.get(0).getId() == getMockedPrices().get(0).getId());
    }

    private List<Price> getMockedPrices(){
        List<Price> mockedPrices = Arrays.asList(new Price(1L, 12.0, 50.12, 12L));
        return mockedPrices;
    }
}

J单元输出:

INFO: 1 * Client response received on thread main
1 < 200
1 < Content-Length: 4
1 < Content-Type: application/json
[{}]


java.lang.AssertionError

调试时:

prices.get(0)
Price
对象,已将
null
分配给所有字段。


更新4:

添加到

configure()

 config.register(JacksonFeature.class);
 config.register(JacksonJsonProvider.class);

现在 Junit 输出好一点了:

INFO: 1 * Client response received on thread main
1 < 200
1 < Content-Length: 149
1 < Content-Type: application/json
[{"id":2,"recurringPrice":122.0,"oneTimePrice":6550.12,"recurringCount":2},{"id":2,"recurringPrice":122.0,"oneTimePrice":6550.12,"recurringCount":2}]

确实列表

prices
有正确的
prices
数量,但所有价格字段都是null。这导致假设问题可能是阅读实体:

List<Price> prices = response.readEntity(new GenericType<List<Price>>(){});

这里是解决方法

将 Moxy 依赖项更改为:

<dependency>
    <groupId>org.glassfish.jersey.media</groupId>
    <artifactId>jersey-media-json-jackson</artifactId>
</dependency>

并在“价格”对象上添加注释。

@XmlRootElement
@JsonIgnoreProperties(ignoreUnknown = true)
java dependency-injection jersey glassfish hk2
4个回答
10
投票

忘记

InjectableProvider
。你不需要它。问题是 mock 服务不是被注入的服务。它是由 DI 框架创建的。所以你正在检查模拟服务的变化,它从未被触及过。

所以你需要做的是将 mock 与 DI 框架绑定。您可以简单地创建另一个

AbstractBinder
进行测试。它可以是一个简单的匿名的,你将在其中绑定 mock

ResourceConfig config = new ResourceConfig(PricingResource.class);
config.register(new AbstractBinder() {
    @Override
    protected void configure() {
        bind(pricingServiceMock).to(PricingService.class);
    }
});

在这里,您只是简单地绑定模拟服务。所以框架会将模拟注入到资源中。现在当您在请求中修改它时,更改将在断言中看到

哦,你仍然需要做你的

when(..).then(..)
来初始化模拟服务中的数据。这也是你所缺少的

@Test
public void testFindPrices(){
    Mockito.when(pricingServiceMock.findSomething()).thenReturn(list);

1
投票

我通过向我的应用程序添加以下依赖项来解决此问题。 编译组:'org.glassfish.jersey.containers.glassfish',名称:'jersey-gf-cdi',版本:'2.14'

那就不需要有任何“AbstractBinder”相关的代码了


1
投票

如果 beans.xml 丢失或放置在错误的位置,也会出现同样的错误。这对我有帮助:beans.xml 应该放在哪里?


0
投票

只需在要注入的类的开头使用

@Service
注释。

import org.jvnet.hk2.annotations.Service

@Service

public class PricingService{....}
© www.soinside.com 2019 - 2024. All rights reserved.