java - InstantiationException 使用 Spring 注入(inject)

标签 java spring tomcat code-injection

我是第一次尝试使用 Spring 注入(inject)。我肯定忘记了一些明显的东西,但我不知道它是什么。

在 src/main/java 下,我有一个包含 Hello、Animal、Cat 的包“example”。

在 src/main/webapp/WEB-INF 下,我有 web.xml 和 springapp-servlet.xml。

当我使用 Tomcat 部署我的应用程序时,我得到:

javax.servlet.ServletException: Error instantiating servlet class example.Hello
    org.apache.catalina.authenticator.AuthenticatorBase.invoke(AuthenticatorBase.java:472)

我缺少什么才能使注入(inject)起作用?

来源如下:

你好.java

package example;

import java.io.IOException;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;

@Service
public class Hello extends HttpServlet {
  private final Animal animal;

  @Autowired
  public Hello(final  Animal animal) {
    this.animal = animal;
  }

  @Override
  protected void doGet(final HttpServletRequest req,
          final HttpServletResponse resp) throws ServletException, IOException {
    resp.getWriter().write(animal.sound());
  }
}

猫.java

package example;

import org.springframework.stereotype.Service;

@Service
public class Cat implements Animal {
  public String sound() {
    return "Miaou";
  }
}

动物.java

package example;

public interface Animal {
  public String sound() ;
}

网络.xml

<?xml version="1.0" encoding="ISO-8859-1"?>
<web-app xmlns="http://java.sun.com/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd"
    version="2.5">

    <context-param>
        <param-name>contextConfigLocation</param-name>
        <param-value>/WEB-INF/springapp-servlet.xml</param-value>
    </context-param>

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

    <servlet>
        <servlet-name>Hello</servlet-name>
        <servlet-class>example.Hello</servlet-class>
    </servlet>
    <servlet-mapping>
        <servlet-name>Hello</servlet-name>
        <url-pattern>/hello</url-pattern>
    </servlet-mapping>

</web-app>

springapp-servlet.xml

<?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:mvc="http://www.springframework.org/schema/mvc"
  xmlns:context="http://www.springframework.org/schema/context"
  xsi:schemaLocation="
        http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-3.0.xsd
        http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
        http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.0.xsd">

  <context:component-scan base-package="example" />
    <mvc:annotation-driven />

</beans>

我最初认为也许我的 springapp-servlet.xml 甚至没有被读取,但是如果我在我的 web.xml 中的名称 springapp-servlet.xml 上打错了,我在部署时确实会出错,所以我显然具有 springapp-servlet.xml 的正确路径。它正在但注入(inject)不起作用。

更新:

由于下面的答案,我在下面展示了对我有用的解决方案。除了 Hello 之外,所有代码都保持不变:

你好.java

public class Hello extends HttpServlet {

  @Inject
  private Animal animal;

  @Override
  public void init(final ServletConfig config) throws ServletException {
    super.init(config);
    SpringBeanAutowiringSupport.processInjectionBasedOnServletContext(this,
            config.getServletContext());
  }

  @Override
  protected void doGet(final HttpServletRequest req,
          final HttpServletResponse resp) throws ServletException, IOException {
    resp.getWriter().write(animal.sound());
  }    
}

最佳答案

这是错误的:

@Service
public class Hello extends HttpServlet {

Servlet 的生命周期是由 servlet 容器控制的,而不是由 Spring 控制的。因此,您不能将 Spring bean 直接 Autowiring 到 servlet。 Spring 根本不应该创建 servlet。基本上 Spring 对您的 servlet 一无所知,它会尝试实例化它,但它与 servlet 容器创建的用于处理请求的实例不同。

最后,您的 servlet 没有无参数构造函数。这样的构造函数是必需的,但它不会使您的示例通过。

解决方案是直接从已注册的 Web 应用程序上下文中获取所需的 Spring bean:

WebApplicationContext context = WebApplicationContextUtils.getRequiredWebApplicationContext(getServletContext());
Animal animal = context.getBean(Animal.class);

另见(其他解决方案)

关于java - InstantiationException 使用 Spring 注入(inject),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14283750/

相关文章:

java - 在 Tomcat 上 curl 部署版本化 war

Java Swing JSplitPane 未按预期显示

java - 如何创建自定义方法以在 Spring 安全表达式语言注释中使用

spring - 连接到本地主机 :5432 refused after docker run when port 5432 is clearly open and listening?

java - REST JAX-RS 日志记录

java - Azure:可以为 Marketplace Tomcat8 做 ARM 模板吗?

java - 如何为 Java App 安装全局键盘处理程序

java - 请求中包含列表的映射对象 - Spring MVC

java - Birt 报告重复行 - 表

java - 在 Spring Rest Controller 中出现 JSON 解析错误 (MismatchedInputException)