向类的侦听器实例发送上下文初始化事件的异常

Posted

技术标签:

【中文标题】向类的侦听器实例发送上下文初始化事件的异常【英文标题】:exception sending context initialized event to listener instance of class 【发布时间】:2013-10-31 02:12:18 【问题描述】:

我有这门课:

public class DatabaseHelper 
    @Autowired
    static UtilService utilService;

    public static void fillEmptyTables()
        if(!isSkillsInDatabase())
            utilService.addGeneratedSkill();
        
        if(!isEventTypesInDatabase())
            utilService.addGeneratedEventType();
        
        if(!isEventStatusesInDatabase())
            utilService.addGeneratedEventStatus();
        

    
    public static boolean isSkillsInDatabase()
        return utilService.getAllSkills().size() != 0; 
    
    public static boolean isEventStatusesInDatabase()
        return utilService.getAllEventStatuses().size() != 0; 
    
    public static boolean isEventTypesInDatabase()
        return utilService.getAllEventTypes().size() != 0; 
    

还有这个听众:

@WebListener
public class ApplicationWebListener implements ServletContextListener 
    @Override
    public void contextDestroyed(ServletContextEvent arg0) 
        System.out.println("ServletContextListener destroyed");
    

    @Override
    public void contextInitialized(ServletContextEvent arg0) 
        System.out.println("ServletContextListener started");
        DatabaseHelper.fillEmptyTables();
    

当我运行我的应用程序时,我看到以下内容:

INFO : org.springframework.web.context.ContextLoader - Root WebApplicationContext: initialization completed in 10029 ms
ServletContextListener started
22.10.2013 13:21:16 org.apache.catalina.core.StandardContext listenerStart
SEVERE: Exception sending context initialized event to listener instance of class com.epam.hhsystem.web.controllers.ApplicationWebListener
java.lang.NullPointerException
    at com.epam.hhsystem.util.DatabaseHelper.isSkillsInDatabase(DatabaseHelper.java:25)
    at com.epam.hhsystem.util.DatabaseHelper.fillEmptyTables(DatabaseHelper.java:13)
    at com.epam.hhsystem.web.controllers.ApplicationWebListener.contextInitialized(ApplicationWebListener.java:19)
    at org.apache.catalina.core.StandardContext.listenerStart(StandardContext.java:4887)
    at org.apache.catalina.core.StandardContext.startInternal(StandardContext.java:5381)
    at org.apache.catalina.util.LifecycleBase.start(LifecycleBase.java:150)
    at org.apache.catalina.core.ContainerBase$StartChild.call(ContainerBase.java:1559)
    at org.apache.catalina.core.ContainerBase$StartChild.call(ContainerBase.java:1549)
    at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:303)
    at java.util.concurrent.FutureTask.run(FutureTask.java:138)
    at java.util.concurrent.ThreadPoolExecutor$Worker.runTask(ThreadPoolExecutor.java:886)
    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:908)
    at java.lang.Thread.run(Thread.java:662)
22.10.2013 13:21:16 org.apache.catalina.core.StandardContext startInternal
SEVERE: Error listenerStart

这个问题的原因是什么?

如何解决?

更新

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

    <!-- DispatcherServlet Context: defines this servlet's request-processing infrastructure -->

    <!-- Enables the Spring MVC @Controller programming model -->
    <annotation-driven />

    <!-- Handles HTTP GET requests for /resources/** by efficiently serving up static resources in the $webappRoot/resources directory -->
    <resources mapping="/resources/**" location="/resources/" />

    <!-- Resolves views selected for rendering by @Controllers to .jsp resources in the /WEB-INF/views directory -->
    <beans:bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
        <beans:property name="prefix" value="/WEB-INF/views/" />
        <beans:property name="suffix" value=".jsp" />
    </beans:bean>

    <context:component-scan base-package="com.epam.hhsystem.web.controllers"/>


</beans:beans>

还有这个:

<import resource="classpath:spring/BeanConfig.xml" />
    <!-- Файл с настройками Security -->
    <import resource="security_config.xml" />

还有这个:

<context:annotation-config />

    <!--ищем тут разные аннотации (например @service)  -->
    <context:component-scan base-package="com.epam.hhsystem.jpa" />
    <context:component-scan base-package="com.epam.hhsystem.util" />
    <context:component-scan base-package="com.epam.hhsystem.services" />

    <context:component-scan base-package="com.epam.hhsystem.web.controllers" />



    <!-- Файл с настройками ресурсов для работы с данными (Data Access Resources) -->
    <import resource="data.xml" />

原来如此。

更新

@WebListener
public class ApplicationWebListener implements ServletContextListener 
    @Override
    public void contextDestroyed(ServletContextEvent arg0) 
        System.out.println("ServletContextListener destroyed");
    

    @Override
    public void contextInitialized(ServletContextEvent arg0) 
        System.out.println("ServletContextListener started");
    

此代码运行良好。当我运行我的应用程序时,我看到了

servletContextListener 已启动

消息

UtilService 位置:

来自配置的字符串:

<context:component-scan base-package="com.epam.hhsystem.services" />

将此代码添加到@Configuration 类后

    @Bean
    ApplicationWebListener getApplicationWebListener()
        return new ApplicationWebListener();
    

我看到旧错误。

更新 2

根据您的建议,我重构了我的代码。我很困惑:

变体1:

   public class MyApplicationListener  implements ApplicationListener<ContextRefreshedEvent> 
            @Autowired
            UtilService utilService;

            public  void fillEmptyTables()
                if(!isSkillsInDatabase())
                    utilService.addGeneratedSkill();
                
                if(!isEventTypesInDatabase())
                    utilService.addGeneratedEventType();
                
                if(!isEventStatusesInDatabase())
                    utilService.addGeneratedEventStatus();
                

            
            public  boolean isSkillsInDatabase()
                return utilService.getAllSkills().size() != 0; 
            
            public  boolean isEventStatusesInDatabase()
                return utilService.getAllEventStatuses().size() != 0; 
            
            public  boolean isEventTypesInDatabase()
                return utilService.getAllEventTypes().size() != 0; 
                

            @Override
            public void onApplicationEvent(ContextRefreshedEvent event) 
                fillEmptyTables();
        


    

这个变体效果很好,但它是架构恐怖

变体 2:

public class MyApplicationListener  implements ApplicationListener<ContextRefreshedEvent> 
                @Override
                public void onApplicationEvent(ContextRefreshedEvent event) 
                           DatabaseHelper.fillEmptyTables();

            

我看到了旧的堆栈跟踪。

为什么?

【问题讨论】:

发布你的spring配置文件 【参考方案1】:

确保您在 Spring 配置文件中启用了组件扫描。 compenent 扫描标记应指向包含UtilService 的包。我想UtilService 目前没有被初始化为 Spring bean,因此它不适合自动装配。

<context:component-scan base-package="package.containing.UtilService" />

尝试在 Spring 配置中将ApplicationWebListener 注册为 bean,我不确定这是否可行,但为了自动装配类中的字段,该类必须是 bean。 Spring 无法将 bean 自动装配到未作为 IOC 容器的一部分实例化的类中。

还有另一种方法。 Spring 提供了一个ApplicationListener 接口,可以由一个类实现,然后注册为一个bean。 ApplicationListener 将在应用程序生命周期的不同时间点被调用,例如启动、重启和关闭。

这是ApplicationListener 的示例:

Listener Example

以及在您的配置文件中注册该监听器的示例,该文件必须是调度程序的配置文件:

Configuration

还有documentation

【讨论】:

我有 @g*** DatabaseHelper 需要是一个在 applicationlistener 中自动装配的 bean Kevin Bowersox 也许我最初的问题是我在静态字段上使用@Autowire? 没有。似乎在ServletContextListener.contextInitialized() 调用所有bean 时都没有自动装配

以上是关于向类的侦听器实例发送上下文初始化事件的异常的主要内容,如果未能解决你的问题,请参考以下文章

向类 org.apache.tomcat.websocket.server.WsContextListener 的侦听器实例发送上下文初始化事件的异常

春天的WebApproot

将 Swagger 添加到 Spring MVC 项目(创建 bean 时出错)

如何为应用程序上下文初始化事件添加钩子?

如何在拖动 React 组件的上下文中移除事件监听器

监听器Listener