代码之家  ›  专栏  ›  技术社区  ›  SanketKD

为什么Spring引导应用程序记录在添加Spring云总线依赖项后启动了两次

  •  3
  • SanketKD  · 技术社区  · 7 年前

    这是我的Spring boot应用程序中的简单代码:

    package com.maxxton.SpringBootHelloWorld;
    
    import org.springframework.boot.SpringApplication;
    import org.springframework.boot.autoconfigure.SpringBootApplication;
    
    @SpringBootApplication
    public class SpringBootHelloWorldApplication {
    
      public static void main(String[] args) {
        SpringApplication.run(SpringBootHelloWorldApplication.class, args);
      }
    }
    

    和一个用于侦听ApplicationEvent的ApplicationListener类:

    package com.maxxton.SpringBootHelloWorld;
    
    import org.springframework.context.ApplicationEvent;
    import org.springframework.context.ApplicationListener;
    import org.springframework.stereotype.Component;
    
    @Component
    public class Test implements ApplicationListener {
    
      @Override
      public void onApplicationEvent(ApplicationEvent event) {
        if (event.getClass().getSimpleName().equals("ApplicationReadyEvent")) {
          System.out.println("-------------------------------------");
          System.out.println(event.getClass().getSimpleName());
          System.out.println("-------------------------------------");
        }
      }
    }
    

    建筑gradle包含以下依赖项:

    dependencies {
    
        compile("org.springframework.boot:spring-boot-starter-amqp")
        compile("org.springframework.cloud:spring-cloud-starter-bus-amqp")
    
        compile('org.springframework.boot:spring-boot-starter-web')
        compile('org.springframework.boot:spring-boot-starter')
        compile("org.springframework.cloud:spring-cloud-starter")
        compile("org.springframework.cloud:spring-cloud-starter-security")
        compile("org.springframework.cloud:spring-cloud-starter-eureka")
        testCompile('org.springframework.boot:spring-boot-starter-test')
    }
    

    [main] c.m.S.SpringBootHelloWorldApplication : Started SpringBootHelloWorldApplication in ... seconds (JVM running for ...)
    

    compile("org.springframework.boot:spring-boot-starter-amqp")
    compile("org.springframework.cloud:spring-cloud-starter-bus-amqp")
    

    这是完整的日志:

    2017-11-17 15:44:07.372  INFO 5976 --- [           main] o.s.c.support.GenericApplicationContext  : Refreshing org.springframework.context.support.GenericApplicationContext@31c7c281: startup date [Fri Nov 17 15:44:07 IST 2017]; root of context hierarchy
    -------------------------------------
    ApplicationReadyEvent
    -------------------------------------
    2017-11-17 15:44:07.403  INFO 5976 --- [           main] c.m.S.SpringBootHelloWorldApplication    : Started SpringBootHelloWorldApplication in 1.19 seconds (JVM running for 10.231)
    2017-11-17 15:44:09.483  WARN 5976 --- [           main] o.s.amqp.rabbit.core.RabbitAdmin         : Failed to declare exchange: Exchange [name=springCloudBus, type=topic, durable=true, autoDelete=false, internal=false, arguments={}], continuing... org.springframework.amqp.AmqpConnectException: java.net.ConnectException: Connection refused: connect
    2017-11-17 15:44:09.492  INFO 5976 --- [           main] o.s.integration.channel.DirectChannel    : Channel 'a-bootiful-client.springCloudBusOutput' has 1 subscriber(s).
    2017-11-17 15:44:09.493  INFO 5976 --- [           main] o.s.c.support.DefaultLifecycleProcessor  : Starting beans in phase 0
    2017-11-17 15:44:09.530  INFO 5976 --- [           main] o.s.i.endpoint.EventDrivenConsumer       : Adding {logging-channel-adapter:_org.springframework.integration.errorLogger} as a subscriber to the 'errorChannel' channel
    2017-11-17 15:44:09.530  INFO 5976 --- [           main] o.s.i.channel.PublishSubscribeChannel    : Channel 'a-bootiful-client.errorChannel' has 1 subscriber(s).
    2017-11-17 15:44:09.530  INFO 5976 --- [           main] o.s.i.endpoint.EventDrivenConsumer       : started _org.springframework.integration.errorLogger
    2017-11-17 15:44:09.530  INFO 5976 --- [           main] o.s.c.support.DefaultLifecycleProcessor  : Starting beans in phase 2147482647
    2017-11-17 15:44:09.539  INFO 5976 --- [           main] c.s.b.r.p.RabbitExchangeQueueProvisioner : declaring queue for inbound: springCloudBus.anonymous.kZ1vvxHaRfChKe1TncH-MQ, bound to: springCloudBus
    2017-11-17 15:44:11.562  WARN 5976 --- [           main] o.s.amqp.rabbit.core.RabbitAdmin         : Failed to declare exchange: Exchange [name=springCloudBus, type=topic, durable=true, autoDelete=false, internal=false, arguments={}], continuing... org.springframework.amqp.AmqpConnectException: java.net.ConnectException: Connection refused: connect
    2017-11-17 15:44:13.587  WARN 5976 --- [           main] o.s.amqp.rabbit.core.RabbitAdmin         : Failed to declare queue: Queue [name=springCloudBus.anonymous.kZ1vvxHaRfChKe1TncH-MQ, durable=false, autoDelete=true, exclusive=true, arguments={}], continuing... org.springframework.amqp.AmqpConnectException: java.net.ConnectException: Connection refused: connect
    2017-11-17 15:44:15.611  WARN 5976 --- [           main] o.s.amqp.rabbit.core.RabbitAdmin         : Failed to declare binding: Binding [destination=springCloudBus.anonymous.kZ1vvxHaRfChKe1TncH-MQ, exchange=springCloudBus, routingKey=#], continuing... org.springframework.amqp.AmqpConnectException: java.net.ConnectException: Connection refused: connect
    2017-11-17 15:44:17.662  INFO 5976 --- [           main] o.s.i.a.i.AmqpInboundChannelAdapter      : started inbound.springCloudBus.anonymous.kZ1vvxHaRfChKe1TncH-MQ
    2017-11-17 15:44:17.662  INFO 5976 --- [           main] o.s.i.endpoint.EventDrivenConsumer       : Adding {message-handler:inbound.springCloudBus.default} as a subscriber to the 'bridge.springCloudBus' channel
    2017-11-17 15:44:17.662  INFO 5976 --- [           main] o.s.i.endpoint.EventDrivenConsumer       : started inbound.springCloudBus.default
    2017-11-17 15:44:17.663  INFO 5976 --- [           main] o.s.c.support.DefaultLifecycleProcessor  : Starting beans in phase 2147483647
    2017-11-17 15:44:17.714  INFO 5976 --- [           main] s.b.c.e.t.TomcatEmbeddedServletContainer : Tomcat started on port(s): 8080 (http)
    -------------------------------------
    ApplicationReadyEvent
    -------------------------------------
    2017-11-17 15:44:17.717  INFO 5976 --- [           main] c.m.S.SpringBootHelloWorldApplication    : Started SpringBootHelloWorldApplication in 20.131 seconds (JVM running for 20.545)
    

    如您所见,ApplicationReadyEvent发生了两次。

    为什么会这样? 有什么方法可以避免这种情况吗?

    2 回复  |  直到 5 年前
        1
  •  7
  •   Gary Russell    7 年前

    您应该让事件侦听器知道它正在运行的应用程序上下文。您还可以使用泛型来选择您感兴趣的事件类型。。。

    @Component
    public class Test implements ApplicationListener<ApplicationReadyEvent>, 
                                 ApplicationContextAware {
    
        private ApplicationContext applicationContext;
    
        @Override
        public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
            this.applicationContext = applicationContext;
        }
    
        @Override
        public void onApplicationEvent(ApplicationReadyEvent event) {
            if (event.getApplicationContext().equals(this.applicationContext)) {
                System.out.println("-------------------------------------");
                System.out.println(event.getClass().getSimpleName());
                System.out.println("-------------------------------------");
            }
        }
    
    }
    
        2
  •  0
  •   Gary Russell    7 年前

    在应用程序中是否使用多个绑定rabbitmq配置。yml/。xml?

    如果是,那么你可以尝试排除RabbitAutoConfiguration。

    @EnableDiscoveryClient
    @EnableAutoConfiguration(exclude = {RabbitAutoConfiguration.class})
    @SpringBootApplication
    public class SpringBootHelloWorldApplication {
    
      public static void main(String[] args) {
        SpringApplication.run(SpringBootHelloWorldApplication.class, args);
      }
    }