Spring Controller中的Init方法(注释版本)


Answers:


238

您可以使用

@PostConstruct
public void init() {
   // ...
}

1
没错,它的“通用注释1.0”,Java1.7也可以使用。
Grim

如果需要使用SecurityContextHolder中的User,则在PostConstruct时刻不进行初始化。需要像无状态方法一样使用它。(getUser()... {return Security ... user();}
Joao Polo

公共或私人
anshulkatta

20

或者,您可以让您的类实现该InitializingBean接口以提供一个回调函数(afterPropertiesSet()),在构造bean时ApplicationContext将调用该函数。


4

有几种方法可以拦截Spring中的初始化过程。如果必须初始化所有bean并自动装配/注入它们,至少我知道有两种方法可以确保这一点。我只有睾丸第二个,但我相信两者都一样。

如果您使用@Bean,则可以通过initMethod进行引用,如下所示。

@Configuration
public class BeanConfiguration {

  @Bean(initMethod="init")
  public BeanA beanA() {
    return new BeanA();
  }
}

public class BeanA {

  // method to be initialized after context is ready
  public void init() {
  }

} 

如果使用@Component,则可以使用@EventListener进行注释。

@Component
public class BeanB {

  @EventListener
  public void onApplicationEvent(ContextRefreshedEvent event) {
  }
}

就我而言,我有一个遗留系统,现在我正在使用IoC / DI,其中Spring Boot是选择的框架。旧系统为表带来了许多循环依赖,因此我必须大量使用setter依赖。这让我有些头疼,因为我无法信任@PostConstruct,因为尚未完成setter的自动装配/注入。顺序是构造函数,@PostConstruct然后是自动绑定的setter。我用@EventListener注释解决了这个问题,该注释将在所有bean的最后一次和“相同”时间运行。该示例还显示了InitializingBean的实现。

我有两个相互依赖的类(@Component)。就本示例而言,这些类看起来相同,仅显示其中一个。

@Component
public class BeanA implements InitializingBean {
  private BeanB beanB;

  public BeanA() {
    log.debug("Created...");
  }

  @PostConstruct
  private void postConstruct() {
    log.debug("@PostConstruct");
  }

  @Autowired
  public void setBeanB(BeanB beanB) {
    log.debug("@Autowired beanB");
    this.beanB = beanB;
  }

  @Override
  public void afterPropertiesSet() throws Exception {
    log.debug("afterPropertiesSet()");
  }

  @EventListener
  public void onApplicationEvent(ContextRefreshedEvent event) {
    log.debug("@EventListener");
  } 
}

这是日志输出,显示了容器启动时调用的顺序。

2018-11-30 18:29:30.504 DEBUG 3624 --- [           main] com.example.demo.BeanA                   : Created...
2018-11-30 18:29:30.509 DEBUG 3624 --- [           main] com.example.demo.BeanB                   : Created...
2018-11-30 18:29:30.517 DEBUG 3624 --- [           main] com.example.demo.BeanB                   : @Autowired beanA
2018-11-30 18:29:30.518 DEBUG 3624 --- [           main] com.example.demo.BeanB                   : @PostConstruct
2018-11-30 18:29:30.518 DEBUG 3624 --- [           main] com.example.demo.BeanB                   : afterPropertiesSet()
2018-11-30 18:29:30.518 DEBUG 3624 --- [           main] com.example.demo.BeanA                   : @Autowired beanB
2018-11-30 18:29:30.518 DEBUG 3624 --- [           main] com.example.demo.BeanA                   : @PostConstruct
2018-11-30 18:29:30.518 DEBUG 3624 --- [           main] com.example.demo.BeanA                   : afterPropertiesSet()
2018-11-30 18:29:30.607 DEBUG 3624 --- [           main] com.example.demo.BeanA                   : @EventListener
2018-11-30 18:29:30.607 DEBUG 3624 --- [           main] com.example.demo.BeanB                   : @EventListener

如您所见,@ EventListener在一切准备就绪并配置好之后最后运行。


-2
public class InitHelloWorld implements BeanPostProcessor {

   public Object postProcessBeforeInitialization(Object bean,
             String beanName) throws BeansException {
       System.out.println("BeforeInitialization : " + beanName);
       return bean;  // you can return any other object as well
   }

   public Object postProcessAfterInitialization(Object bean,
             String beanName) throws BeansException {
       System.out.println("AfterInitialization : " + beanName);
       return bean;  // you can return any other object as well
   }

}
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.