一尘不染

使用scheduler- BeanCreationNotAllowedException进行Spring引导:创建名称为'entityManagerFactory'的bean时出错:不允许创建单例bean

spring-boot

我们有一个带有调度程序的spring boot项目,它以固定的时间间隔从数据库中读取数据。

在使用maven从STS构建项目的过程中,即使最终构建状态成功 ,在运行测试用例 的控制台中 我们也将遇到错误。

org.springframework.beans.factory.BeanCreationNotAllowedException:创建名称为’entityManagerFactory’的bean时出错:在破坏该工厂的singleton时,不允许创建singleton
bean(请勿在destroy方法实现中向BeanFactory请求bean!)
org.springframework.beans.factory.support.DefaultSingletonBeanRegistry.getSingleton(DefaultSingletonBeanRegistry.java:216)在org.springframework.beans.factory.support.AbstractBeanFactory.doGetBean(AbstractBeanFactory.java:299)在org.springframework.beans.factory。
org.springframework.beans.factory.BeanFactoryUtils上的support.AbstractBeanFactory.getBean(AbstractBeanFactory.java:199)在org.springframework.beans.factory.support.DefaultListableBeanFactory.getBeansOfType(DefaultListableBeanFactory.java:523)org.springframeframe.dao.support.PersistenceExceptionTranslationInterceptor.invoke(PersistenceExceptionTranslationInterceptor.java:145)上的beansOfTypeIn
includedAncestors(BeanFactoryUtils.java:276)位于org.springframework.dao.support.PersistenceExceptionTranslationInterceptor.detectPersistenceExceptionTranslators(PersistenceExceptionTranslationInterceptor.java:162)在org.springframework.org.org.springframework.data.jpa.repository.support处的.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:179)在org.springframework处的org.springframework.data.jpa.repository.support.CrudMethodMetadataPostProcessor
$
CrudMethodMetadataPopulatingMethodIntercceptor.invoke(CrudMethodMetadataPostReflectop.java:122)
org.springframework.aop.interceptor上的.proceed(ReflectiveMethodInvocation.java:179)。在org.springframework.aop.framework.JdkDynamicAopProxy.invoke(JdkDynamicAopProxy.org.org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:179)处的ExposeInvocationInterceptor.invoke(ExposeInvocationInterceptor.java:92)。
.sun.proxy。$ Proxy70.findByTraIdAndTransactionNameAndExecutionTime(未知来源)位于

申请文件

@SpringBootApplication
@PropertySource("classpath:application.properties")
@EnableScheduling
public class ProvisioningApplication {

    public static void main(String[] args) {

        SpringApplication.run(ProvisioningApplication.class, args);

    }
}

调度程序文件

BusinessService具有读取数据库的逻辑

@Component
public class SchedulerJob {

    @Autowired
    BusinessService service;

    @Scheduled(fixedRate=300000) //5mnts
    public void schdeule() {
        service.startService();

    }
}

测试文件

@RunWith(SpringJUnit4ClassRunner.class)
@SpringApplicationConfiguration(classes = ProvisioningApplication.class)
public class ProvisioningApplicationTests {

    @Test
    public void contextLoads() {
    }

}

这里的问题是,为什么Spring Boot在构建项目时运行调度程序任务,为什么它会引发上述异常?


阅读 831

收藏
2020-05-30

共1个答案

一尘不染

Spring Boot中 ,当您进行Maven构建时,默认情况下会运行测试用例。在这种情况下, 运行 集成测试
脚本,它将尝试连接到数据库。由于您没有任何要作为项目集成测试一部分的内容。一种可能的解决方案是将您的
ProvisioningApplicationTests 类声明为 abstract 。这将限制对
ProvisioningApplicationTests 类的实例创建。

@RunWith(SpringJUnit4ClassRunner.class)
@SpringApplicationConfiguration(classes = ProvisioningApplication.class)
public abstract class ProvisioningApplicationTests {
    @Test
    public void contextLoads() {
    }
  }

解决此问题的另一种方法是在pom.xml中包含以下代码

<plugins>
   <plugin>
    <artifactId>maven-surefire-plugin</artifactId>
    <configuration>
     <skipTests>false</skipTests>
     <excludes>
      <exclude>**/*IT.java</exclude>
     </excludes>
    </configuration>
   </plugin>
   <plugin>
    <artifactId>maven-failsafe-plugin</artifactId>
    <executions>
     <execution>
      <id>integration-test</id>
      <goals>
       <goal>integration-test</goal>
      </goals>
      <configuration>
       <skipTests>true</skipTests>
       <includes>
        <include>**/*IT.class</include>
       </includes>
      </configuration>
     </execution>
    </executions>
   </plugin>
  </plugins>

这将排除在构建项目时要执行的集成测试类。 maven-surefire-plugin 用于运行单元测试。 maven-failsafe-
plugin
用于运行集成测试。使用这种方法时,请确保所有集成类文件名都以 ‘IT’ 结尾。例如UserTestIT.java

2020-05-30