Bean Lifecycle in Spring - Advanced Guide

1️⃣ Introduction

The Spring bean lifecycle consists of several phases from instantiation to destruction. This article explores how to understand and manage these phases effectively.

Key features include:

  • Bean instantiation
  • Dependency injection
  • Initialization callbacks
  • Destruction callbacks
  • Lifecycle interfaces

2️⃣ Key Concepts & Terminology

  • Instantiation: Creating the bean instance
  • Population: Setting properties and dependencies
  • Initialization: Setting up the bean after dependency injection
  • Destruction: Cleaning up resources before bean removal
  • Lifecycle Callbacks: Methods called at specific lifecycle phases

3️⃣ Hands-on Implementation 🛠

🔹 Step 1: Using @PostConstruct and @PreDestroy

@Service
public class DatabaseService {
    private Connection connection;
    
    @PostConstruct
    public void init() {
        connection = createConnection();
        System.out.println("Database connection established");
    }
    
    @PreDestroy
    public void cleanup() {
        closeConnection(connection);
        System.out.println("Database connection closed");
    }
}

🔹 Step 2: Implementing InitializingBean and DisposableBean

@Service
public class CacheService implements InitializingBean, DisposableBean {
    private Map cache;
    
    @Override
    public void afterPropertiesSet() {
        cache = new HashMap<>();
        System.out.println("Cache initialized");
    }
    
    @Override
    public void destroy() {
        cache.clear();
        System.out.println("Cache cleared");
    }
}

🔹 Step 3: Using Bean Post Processors

@Component
public class LoggingBeanPostProcessor implements BeanPostProcessor {
    @Override
    public Object postProcessBeforeInitialization(Object bean, String beanName) {
        System.out.println("Before initialization of " + beanName);
        return bean;
    }
    
    @Override
    public Object postProcessAfterInitialization(Object bean, String beanName) {
        System.out.println("After initialization of " + beanName);
        return bean;
    }
}

4️⃣ Common Issues & Debugging 🐞

Common Issues and Solutions

Issue Solution
Initialization order issues Use @DependsOn or implement Ordered interface
Resource cleanup not happening Ensure proper shutdown hook registration
Post processor not working Check bean registration order

5️⃣ Q&A / Frequently Asked Questions

The order is: 1) Instantiation, 2) Population of properties, 3) BeanNameAware, 4) BeanFactoryAware, 5) ApplicationContextAware, 6) Pre-initialization (BeanPostProcessor), 7) Initialization (@PostConstruct), 8) After-initialization (BeanPostProcessor), 9) Ready for use, 10) Destruction (@PreDestroy).

Use @PostConstruct for simple initialization tasks as it's more concise and decoupled from Spring. Use InitializingBean when you need more control over the initialization process or need to implement multiple lifecycle interfaces.

6️⃣ Best Practices & Pro Tips 🚀

  • Use @PostConstruct and @PreDestroy for simple lifecycle management
  • Implement lifecycle interfaces for complex scenarios
  • Use BeanPostProcessor for cross-cutting concerns
  • Handle resource cleanup properly
  • Consider initialization order
  • Document lifecycle dependencies

7️⃣ Read Next 📖

8️⃣ Conclusion

Understanding and properly managing bean lifecycle is crucial for building robust Spring applications. Choose the appropriate lifecycle management approach based on your application's needs.

Remember to handle resource cleanup properly and consider the order of initialization when beans depend on each other.