Spring 2.5 features a new pointcut designator – bean() that allows selecting join points in beans with a matching name pattern. Now it is possible to use the auto-proxy mechanism along with Spring-AspectJ integration to select a specific bean even when there are more than one beans of a type. Earlier, you could use BeanNameAutoProxyCreator to achieve a similar result; however, that mechanism didn’t work with Schema-style or @AspectJ aspects.
Besides selecting a specific bean, this pointcut designator offers two interesting ways to select beans if you follow an appropriate naming convention:
- Selecting a vertical slice of beans: If you follow a convention where bean names include a string indicating their role from the business perspective, a bean() pointcut can select beans based on their business role. For example, you may use the bean(account*) pointcut to select all accounting-related beans such as accountRepository, accountService, and accountController if bean names start with a string representing their business functionality.
- Selecting a horizontal slice of beans: If you follow a convention where bean names include a string indicating their role from the architectural perspective, a bean() pointcut can select beans based on their architectural role. For example, you can use bean(*Repository) to select all repository beans if bean names end with a string representing their architectural role. Without the bean() pointcut, you had to rely on the package structure or type-based pointcuts, which can be sometimes a bit too restrictive.
Figure 1: Selecting horizontal and vertical slices of beans based on their names using bean() pointcuts
This pointcut represents a Spring-specific extension to the AspectJ pointcut expression language and as such is useful only in a Spring-based application. The name-pattern follows the AspectJ matching rules for a name pattern with ’’ being the only allowed wildcard. Here is a table showing a few example pointcuts and beans selected by them.
|Pointcut||Join points selected in|
|bean(accountRepository)||The bean named “accountRepository”|
|!bean(accountRepository)||Any bean except the “accountRepository” bean|
|bean(account*)||Any bean with name starting in “account”|
|bean(*Repository)||Any bean with name ending in “Repository”|
|bean(accounting/showaccount)||The bean named accounting/showaccount (designating, say, a controller handling that URL)|
|bean(accounting/*)||Any bean whose name starts with “accounting/” (designating, say, any controller handling accounting-related URLs)|
|bean(accounting/*/edit)||Any bean whose name starts with “accounting/” and ends with “/edit” (designating, say, any controller handling the edit operation functionality related to accounting)|
|bean(*dataSource) || bean(*DataSource)||Any bean whose name ends with either “dataSource” or “DataSource”|
|bean(service:name=monitoring)||The bean named “service:name=monitoring”|