Spring DI - свойство Autowired имеет значение null в службе REST

Я начинаю с Spring DI, но я борюсь с инъекцией зависимостей, а хуже всего то, что я даже не уверен, почему, как мне кажется. Надеюсь, вы, ребята, можете мне помочь!

Проблема заключается в том, что свойство, аннотированное как @Autowired, всегда имеет значение null

У меня есть несколько проектов с структурой Maven:

  • com.diegotutor.lessondeliver
  • com.diegotutor.utility

Я запускаю примеры над Tomcat 7

Я использую следующие зависимости в своем pom.xml:

  • spring -context 3.2.4
  • spring -web 3.2.4
  • jersey-server 1.17.1
  • джерси-ядро 1.17.1
  • jersey-servlet 1.17.1

Простая идея состоит в том, чтобы иметь службу RESTful, которая через Dependency Injection может распечатать значение свойства, расположенного в файле конфигурации, расположенном по адресу: D:\configuracion.conf.

В com.diegotutor.utility У меня есть следующий интерфейс:

package com.diegotutor.utility;

public interface ConfigService {

    public String getProperty(final String propertyName);
}

Реализовано:

package com.diegotutor.utility.impl;

import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.Reader;
import java.util.Properties;

import com.diegotutor.utility.ConfigService;

public class PropertyFileConfigService implements ConfigService{

Properties prop;

public PropertyFileConfigService (final InputStream input) throws IOException {
    if(input == null) {
        throw new IllegalArgumentException("Input stream can't be null");
    }
    prop = new Properties();
    prop.load(input);
}

public PropertyFileConfigService (final String fileName) throws IOException {
    final FileInputStream input = new FileInputStream(fileName);
    prop = new Properties();
    prop.load(input);
}

public PropertyFileConfigService(final Reader input) throws IOException {
    prop = new Properties();
    prop.load(input);
}

public String getProperty(final String propertyName) {
    return prop.getProperty(propertyName);
}

}

И в com.diegotutor.lessondeliver У меня есть служба RESTful, где я хотел бы использовать инъецируемый экземпляр ConfigService:

package com.diegotutor.lessondeliver;

import javax.ws.rs.GET;
import javax.ws.rs.Path;
import javax.ws.rs.Produces;
import javax.ws.rs.core.MediaType;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

import com.diegotutor.utility.ConfigService;

@Path("/")
@Component
public class HelloWorld {

private static final Log log = LogFactory.getLog(HelloWorld.class);

@Autowired
private ConfigService configService;

@Path("/helloworld")
@GET
@Produces(MediaType.TEXT_PLAIN)
public String getHello() {
    String host = configService.getProperty("host");
    return "Hello World! HOST" + host;
            // configService IS NULL!! 
            //SO IT THROWS A NULLPOINTER EXCEPTION WHEN INVOKING getProperty ON IT
}
}

Наконец, в /com.diegotutor.lessondeliver/src/main/webapp/WEB-INF/service-beans.xml у меня есть следующий файл контекста XML-приложения, где я использую реализацию ConfigService (PropertyFileConfigService), введя на нем путь для файла конфигурации, который нужно прочитать:

<?xml version="1.0" encoding="UTF-8"?>
 <beans xmlns="http://www.springframework.org/schema/beans"
 xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  xmlns:context="http://www.springframework.org/schema/context"
  xsi:schemaLocation="http://www.springframework.org/schema/beans
   http://www.springframework.org/schema/beans/spring-beans.xsd
   http://www.springframework.org/schema/context
   http://www.springframework.org/schema/context/spring-context.xsd">

<bean id="configService" class="com.diegotutor.utility.impl.PropertyFileConfigService">
    <constructor-arg type="java.lang.String"
        value="D:\configuracion.conf" />
</bean>

<context:component-scan base-package="com.diegotutor" />

</beans>

Очевидно, что я указал в web.xml этого веб-приложения com.diegotutor.lessondeliver, что хочу service-beans.xml как ConfigLocation и прослушиватель ContextLoaderListener, а служба RESTful полагается на ServletContainer

Если я задаю контекст: компонент-сканирование, чтобы искать Компоненты в com.diegotutor, как предложено здесь, и я заставляю создание объекта через Spring by не используя какое-либо новое выражение, как предлагалось здесь, Почему я получаю аннотированный configService как null? Почему Spring не может ввести экземпляр com.diegotutor.utility.impl.PropertyFileConfigService?

Любая помощь будет высоко оценена!

Спасибо

Редакция:

В соответствии с запросом мой web.xml выглядит следующим образом:

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
xmlns="http://java.sun.com/xml/ns/javaee" 
xmlns:web="http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" 
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee     http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd" 
id="WebApp_ID" version="3.0">

 <display-name>com.diegotutor.lessondeliver</display-name>

 <context-param>
    <param-name>contextConfigLocation</param-name>
    <param-value>/WEB-INF/service-beans.xml</param-value>
 </context-param>

 <listener>
    <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
 </listener>

 <servlet>
  <servlet-name>jersey-servlet</servlet-name>
  <servlet-class>
         com.sun.jersey.spi.container.servlet.ServletContainer
  </servlet-class>
  <load-on-startup>1</load-on-startup>
 </servlet>

 <servlet-mapping>
  <servlet-name>jersey-servlet</servlet-name>
  <url-pattern>/rest/*</url-pattern>
 </servlet-mapping>
</web-app>

Ответы

Ответ 1

Вы были правы! Похоже, что проблема в том, что Джерси полностью не знает о Spring и создает свой собственный объект. Чтобы сделать Джерси осведомленным о создании Spring объектов (через инъекцию зависимостей), мне пришлось интегрировать Spring + Jersey.

Для интеграции:

  • Добавить зависимости maven

    <dependency>
      <groupId>com.sun.jersey.contribs</groupId>
      <artifactId>jersey-spring</artifactId>
      <version>1.17.1</version>
      <exclusions>
        <exclusion>
          <groupId>org.springframework</groupId>
          <artifactId>spring-core</artifactId>
        </exclusion>
        <exclusion>
          <groupId>org.springframework</groupId>
          <artifactId>spring-beans</artifactId>
        </exclusion>
        <exclusion>
          <groupId>org.springframework</groupId>
          <artifactId>spring-context</artifactId>
        </exclusion>
        <exclusion>
          <groupId>org.springframework</groupId>
          <artifactId>spring-web</artifactId>
        </exclusion>
      </exclusions>
    </dependency>
    
    • Используйте SpringServlet для jersey-servlet в web.xml

       <servlet>
        <servlet-name>jersey-servlet</servlet-name>
        <servlet-class>
               com.sun.jersey.spi.spring.container.servlet.SpringServlet
        </servlet-class>
        <load-on-startup>1</load-on-startup>
       </servlet>
      

Теперь @Autowired работает правильно, и объект больше не равен нулю.

Я немного смущен об исключениях, которые я должен использовать в maven при использовании jersey- spring зависимости, но это еще одна проблема:)

Спасибо!

Ответ 2

Интеграция Spring с Джерси 2 (org.glassfish.*):

Maven

Некоторые зависимости могут быть ненужными, проверьте и очистите их после того, как все стало работать.

    <properties>
        <jersey.version>2.5</jersey.version>
    </properties>

    <!-- Jersey -->
    <dependency>
        <groupId>org.glassfish.jersey.core</groupId>
        <artifactId>jersey-server</artifactId>
        <version>${jersey.version}</version>
    </dependency>

    <dependency>
        <groupId>org.glassfish.jersey.containers</groupId>
        <!-- if your container implements Servlet API older than 3.0, use "jersey-container-servlet-core" -->
        <artifactId>jersey-container-servlet</artifactId>
        <version>${jersey.version}</version>
    </dependency>
    <dependency>
        <groupId>org.glassfish.jersey.core</groupId>
        <artifactId>jersey-client</artifactId>
        <version>${jersey.version}</version>
    </dependency>
    <dependency>
        <groupId>org.glassfish.jersey.test-framework.providers</groupId>
        <artifactId>jersey-test-framework-provider-inmemory</artifactId>
        <version>${jersey.version}</version>
    </dependency>

    <!-- Jersey + Spring -->
    <dependency>
        <groupId>org.glassfish.jersey.ext</groupId>
        <artifactId>jersey-spring3</artifactId>
        <version>${jersey.version}</version>
        <exclusions>
            <exclusion>
                <groupId>org.springframework</groupId>
                <artifactId>spring</artifactId>
            </exclusion>
            <exclusion>
                <groupId>org.springframework</groupId>
                <artifactId>spring-core</artifactId>
            </exclusion>
            <exclusion>
                <groupId>org.springframework</groupId>
                <artifactId>spring-web</artifactId>
            </exclusion>
            <exclusion>
                <groupId>org.springframework</groupId>
                <artifactId>spring-beans</artifactId>
            </exclusion>
            <exclusion>
                <groupId>org.springframework</groupId>
                <artifactId>spring-context</artifactId>
            </exclusion>
        </exclusions>
    </dependency>

web.xml

<servlet>
    <servlet-name>my-rest-service</servlet-name>
    <servlet-class>org.glassfish.jersey.servlet.ServletContainer</servlet-class>
    <init-param>
        <param-name>jersey.config.server.provider.packages</param-name>
        <param-value>my.package.with.rest.services</param-value>
    </init-param>
    <load-on-startup>1</load-on-startup>
</servlet>

<servlet-mapping>
    <servlet-name>my-rest-service</servlet-name>
    <url-pattern>/api/*</url-pattern>
</servlet-mapping>

applicationContext.xml

Во время обновления Spring мне пришлось переместить его с /main/webapp/WEB-INF/ на /main/resources/ (подробнее).

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context"
    xsi:schemaLocation="http://www.springframework.org/schema/beans
    http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
    http://www.springframework.org/schema/context
    http://www.springframework.org/schema/context/spring-context-2.5.xsd">

    <context:annotation-config />
    <context:component-scan base-package="my.package.with.rest.services" />
</beans>

Пример службы REST

public interface MyService
{
    String work(String s);
}

...

@Service
public class MyServiceImpl implements MyService
{
    @Override
    public String work(String s)
    {
        return "Hello, " + s;
    }
}

...

@Path("demo/")
@Component
public class DemoRestService
{
    @Autowired
    private MyService service;

    @GET
    @Path("test")
    public Response test(@FormParam("param") String par)
    {
        try
        {
            String entity = service.work(par);
            return Response.ok(entity).build();
        }
        catch (Exception e)
        {
            e.printStackTrace();
            return Response.status(Status.INTERNAL_SERVER_ERROR).entity("Epic REST Failure").build();
        }
    }
}

Ответ 3

Еще одна возможная опция - вручную активировать autowiring в ресурсе трикотажа:

@Context
private ServletContext servletContext;

@PostConstruct
public void init() {
    SpringBeanAutowiringSupport.processInjectionBasedOnServletContext(this, servletContext);
}

Хмм, вы получаете "ручное автоуправление"...

Ответ 4

или вы можете просто расширить класс SpringBeanAutoWiringSupport. Например: public class DemoRestService extends SpringBeanAutoWiringSupport. Расширяя этот класс поддержки, свойства вашего класса обслуживания могут быть автоматически подключены.