Using Spring Aware Interfaces

Using Spring Aware Interfaces

2 Comments

Spring Aware Interfaces allow you to hook into the inner workings of the Spring Framework. Though Spring Aware interfaces, you can access the Spring context, or Spring Bean life cycle events.

Your Spring beans might require access to framework objects, such as ApplicationContext, BeanFactory, and ResourceLoader . To gain access, a bean can implement one of the many Aware interfaces of the Spring Framework.

When a bean implements an Aware interface, the Spring Framework injects a particular framework object to the bean through a callback-style method. The object Spring injects depends on the interface which the bean implements. For example, if the bean implements the ApplicationContextAware interface, Spring will inject an ApplicationContext object in to the bean.

In this post, we will learn about the Spring aware interfaces, particularly ApplicationContextAware, BeanFactoryAware, and BeanNameAware interfaces.

In the bean lifecycle, the Spring Framework calls the aware interface methods after populating bean properties and just before pre initialization with BeanPostProcessor.

Aware interfaces Callbacks in Bean Lifecycle

The ApplicationContextAware Interface

In Spring beans, you might require access to the ApplicationContext. For example, if your bean needs to look up some other beans. Similarly, if your bean needs access to some application file resource in your bean or even publish some application events, you need access to the ApplicationContext.

Spring provides an ApplicationContextAware interface that allows beans access to the ApplicationContext. This interface provides a single setApplicationContext method.

void setApplicationContext(ApplicationContext applicationContext)
throws BeansException

The following code shows the use of ApplicationContextAware.

package guru.springframework.springawaredemo.awareimpls;

import guru.springframework.springawaredemo.domain.User;
import org.springframework.beans.BeansException;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
public class ApplicationContextAwareImpl implements ApplicationContextAware {
@Override
public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
User user = (User) applicationContext.getBean("user");
System.out.println("User Id: " + user.getUserId() + " User Name :" + user.getName());
}
}

The preceding code is of a bean that implements ApplicationContextAware. The code overrides the setApplicationContext() method to lookup another bean with the id user using the injected ApplicationContext.

The BeanFactoryAware Interface

Beans might need access to the bean factory that created it, say to call any service from the bean factory.
Should you need to obtain a reference to the bean factory, implement the BeanFactoryAware interface. This interface provides the setBeanFactory() method.

void setBeanFactory(BeanFactory beanFactory)

The preceding setBeanFactory() is a callback that supplies the owning factory to the bean instance.
Here is an example of a bean that implements the BeanFactoryAware Interface.

package guru.springframework.springawaredemo.awareimpls;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.BeanFactory;
import org.springframework.beans.factory.BeanFactoryAware;
public class BeanFactoryAwareImpl implements BeanFactoryAware {
@Override
public void setBeanFactory(BeanFactory beanFactory) throws BeansException {
System.out.println(beanFactory.getBean("user"));
}
}

The BeanNameAware Interface

The BeanNameAware interface is implemented by beans that need access to its name defined in the Spring container. This interface provides the setBeanName() method.

void setBeanFactory(BeanFactory beanFactory)

The preceding setBeanFactory() is a callback that supplies the name of the bean.

Here is an example of a bean which implements the BeanNameAware Interface.

package guru.springframework.springawaredemo.awareimpls;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.BeanFactory;
import org.springframework.beans.factory.BeanFactoryAware;
public class BeanFactoryAwareImpl implements BeanFactoryAware {
@Override
public void setBeanFactory(BeanFactory beanFactory) throws BeansException {
System.out.println(beanFactory.getBean("user"));
}
}

Summary

Although I covered only three, there are additional aware interfaces. The Aware API documentation provides complete details.

One interface I would like to specially mention is the ResourceLoaderAware interface. Implement this interface if a bean needs to load resources present in the classpath or file system. On implementing ResourceLoaderAware, your bean is notified of the ResourceLoader (typically the ApplicationContext) that it runs in. This is an alternative to a full ApplicationContext dependency via the ApplicationContextAware interface.

I’ll be writing a detailed post on ResourceLoaderAware that you can check out.

As an end note, I’m not particularly fond of the aware interfaces. Implementing them ties your application to the Spring framework, thus inverting the whole inversion-of-control concept. In an ideal world, your application should not be aware of being managed by an ApplicationContext at all or tied to any framework objects.

Also, note that ApplicationContextAware is the legacy version that has been around at least since Version 2.0. @Autowired ApplicationContext applicationContext and @Inject ApplicationContext applicationContext are the newer mechanisms but they work in pretty much the same way. Although the newer ones require lesser typing of code, I’d probably go with ApplicationContextAware, because it semantically makes clear what it is about.

The source code for this post can be found here on GitHub.

About SFG Contributor

Staff writer account for Spring Framework Guru

    You May Also Like

    2 comments on “Using Spring Aware Interfaces

    1. August 30, 2019 at 1:26 pm

      Great explainer, as always!
      Just a little comment.. the BeanNameAware example looks very similar to the one for BeanFactoryAware.

      Reply
    2. September 6, 2019 at 11:18 am

      In the section “The BeanNameAware Interface” you should implement BeanNameAware not BeanFactoryAware and consequently you modifiy the method.

      Reply

    Leave a Reply

    Your email address will not be published. Required fields are marked *

    This site uses Akismet to reduce spam. Learn how your comment data is processed.