Spring: Beans, Components, Services
Component declaration
Simple annotation
@Component
public class PageInfo
{
}
Annotation with specified scope
@Component
@RequestScope
public class PageInfo
{
}
Bean declaration
To declare a Bean object, it is enough to specify the @Bean annotation to the method that returns the bean type both in classes with the @Configuration annotation and in classes with the @Component annotation (or its descendants).
@Configuration
public class LessonsConfiguration
{
@Bean
GreetingService greetingService() {
return new GreetingServiceImpl();
}
}
Named beans
@Bean(name = "gServiceName")
@Bean(name = {"gServiceName", "gServiceAnotherNamed"})
Bean scopes
https://docs.spring.io/spring-framework/reference/core/beans/factory-scopes.html
Scope | Description |
---|---|
singleton | (Default) Scopes a single bean definition to a single object instance for each Spring IoC container. |
prototype | Scopes a single bean definition to any number of object instances. |
request | Scopes a single bean definition to the lifecycle of a single HTTP request. That is, each HTTP request has its own instance of a bean created off the back of a single bean definition. Only valid in the context of a web-aware Spring ApplicationContext. |
session | Scopes a single bean definition to the lifecycle of an HTTP Session. Only valid in the context of a web-aware Spring ApplicationContext. |
application | Scopes a single bean definition to the lifecycle of a ServletContext. Only valid in the context of a web-aware Spring ApplicationContext. |
websocket | Scopes a single bean definition to the lifecycle of a WebSocket. Only valid in the context of a web-aware Spring ApplicationContext. |
{#beans-factory-scopes-tbl}
@Bean vs @Component
@Component is a class level annotation whereas @Bean is a method level annotation and name of the method serves as the bean name
We cannot create a bean of a class using @Component, if the class is outside spring container whereas we can create a bean of a class using @Bean even if the class is present outside the spring container. By using the @Bean annotation, you can wrap a third-party class (it may not have @Component and it may not use Spring), as a Spring bean.
https://stackoverflow.com/questions/10604298/spring-component-versus-bean
@Configuration
@Configuration is a class-level annotation indicating that an object is a source of bean definitions. @Configuration classes declare beans through @Bean-annotated methods. Calls to @Bean methods on @Configuration classes can also be used to define inter-bean dependencies.
@ComponentScan
Official documentation states: We generally recommend that you locate your main application class in a root package above other classes.
So that your configuration can know about such components and you can use them, there is a special annotation for your configuration class @ComponentScan. By default, this configuration scans for classes with the @Component annotation and its descendants in the package in which it is located, as well as in subpackages.
However, if you want the scan to be for specific directories, then this can be configured by simply adding the basePackages parameter to the @ComponentScan annotation, indicating one or more packages. It will look something like this: @ComponentScan(basePackages = "lessons.services")
@ComponentScan without arguments tells Spring to scan the current package and all of its sub-packages.
You can use aproach as in example: ( https://stackoverflow.com/questions/55750957/why-is-this-componentscan-not-working-in-my-spring-boot-application )
@ComponentScan
@ComponentScan(basePackages = {"com.dionext"})
@EnableJpaRepositories(basePackages = "com.dionext")
@EntityScan(basePackages = "com.dionext")
public class ExFrontendDemoApplication {
@Component @Repository, @Service и @Controller
Annotation | Meaning |
---|---|
@Component | generic stereotype for any Spring-managed component |
@Repository | stereotype for persistence layer |
@Service | stereotype for service layer |
@Controller | stereotype for presentation layer (spring-mvc) |
In addition to pointing out, that this is an Annotation based Configuration, @Repository's job is to catch platform specific exceptions and re-throw them as one of Spring's unified unchecked exception.
The dispatcher scans the classes annotated with @Controller and detects methods annotated with @RequestMapping annotations within them. We can use @RequestMapping on/in only those methods whose classes are annotated with @Controller
Pre and Post Events
Use annotation: @PostConstruct and @PreDestroy
Bean access {#scopes}
Example
public class SomeClass {
private ApplicationContext context;
@Autowired
public void setContext(ApplicationContext context) { this.context = context; }
public void someMethod(){
GreetingService greetingService = context.getBean(GreetingService.class);
}
}
https://www.baeldung.com/spring-get-current-applicationcontext
Autowiring
Recommended:
Constructor-based Dependency Injection
public class SimpleMovieLister {
// the SimpleMovieLister has a dependency on a MovieFinder
private final MovieFinder movieFinder;
// a constructor so that the Spring container can inject a MovieFinder
public SimpleMovieLister(MovieFinder movieFinder) {
this.movieFinder = movieFinder;
}
// business logic that actually uses the injected MovieFinder is omitted...
}
Setter-based Dependency Injection
public class SimpleMovieLister {
// the SimpleMovieLister has a dependency on the MovieFinder
private MovieFinder movieFinder;
// a setter method so that the Spring container can inject a MovieFinder
public void setMovieFinder(MovieFinder movieFinder) {
this.movieFinder = movieFinder;
}
// business logic that actually uses the injected MovieFinder is omitted...
}