public class DefaultSingletonBeanRegistry extends SimpleAliasRegistry implements SingletonBeanRegistry
SingletonBeanRegistry
.
Allows for registering singleton instances that should be shared
for all callers of the registry, to be obtained via bean name.
Also supports registration of
DisposableBean
instances,
(which might or might not correspond to registered singletons),
to be destroyed on shutdown of the registry. Dependencies between
beans can be registered to enforce an appropriate shutdown order.
This class mainly serves as base class for
BeanFactory
implementations,
factoring out the common management of singleton bean instances. Note that
the ConfigurableBeanFactory
interface extends the SingletonBeanRegistry
interface.
Note that this class assumes neither a bean definition concept
nor a specific creation process for bean instances, in contrast to
AbstractBeanFactory
and DefaultListableBeanFactory
(which inherit from it). Can alternatively also be used as a nested
helper to delegate to.
registerSingleton(java.lang.String, java.lang.Object)
,
registerDisposableBean(java.lang.String, org.springframework.beans.factory.DisposableBean)
,
DisposableBean
,
ConfigurableBeanFactory
Modifier and Type | Field and Description |
---|---|
private java.util.Map<java.lang.String,java.util.Set<java.lang.String>> |
containedBeanMap
Map between containing bean names: bean name --> Set of bean names that the bean contains
|
private java.util.Map<java.lang.String,java.util.Set<java.lang.String>> |
dependenciesForBeanMap
Map between depending bean names: bean name --> Set of bean names for the bean's dependencies
|
private java.util.Map<java.lang.String,java.util.Set<java.lang.String>> |
dependentBeanMap
Map between dependent bean names: bean name --> Set of dependent bean names
|
private java.util.Map<java.lang.String,java.lang.Object> |
disposableBeans
Disposable bean instances: bean name --> disposable instance
|
private java.util.Map<java.lang.String,java.lang.Object> |
earlySingletonObjects
Cache of early singleton objects: bean name --> bean instance
|
private java.util.Set<java.lang.String> |
inCreationCheckExclusions
Names of beans currently excluded from in creation checks
|
protected Log |
logger
Logger available to subclasses
|
protected static java.lang.Object |
NULL_OBJECT
Internal marker for a null singleton object:
used as marker value for concurrent Maps (which don't support null values).
|
private java.util.Set<java.lang.String> |
registeredSingletons
Set of registered singletons, containing the bean names in registration order
|
private java.util.Map<java.lang.String,ObjectFactory<?>> |
singletonFactories
Cache of singleton factories: bean name --> ObjectFactory
|
private java.util.Map<java.lang.String,java.lang.Object> |
singletonObjects
Cache of singleton objects: bean name --> bean instance
|
private java.util.Set<java.lang.String> |
singletonsCurrentlyInCreation
Names of beans that are currently in creation
|
private boolean |
singletonsCurrentlyInDestruction
Flag that indicates whether we're currently within destroySingletons
|
private java.util.Set<java.lang.Exception> |
suppressedExceptions
List of suppressed Exceptions, available for associating related causes
|
Constructor and Description |
---|
DefaultSingletonBeanRegistry() |
Modifier and Type | Method and Description |
---|---|
protected void |
addSingleton(java.lang.String beanName,
java.lang.Object singletonObject)
Add the given singleton object to the singleton cache of this factory.
|
protected void |
addSingletonFactory(java.lang.String beanName,
ObjectFactory<?> singletonFactory)
Add the given singleton factory for building the specified singleton
if necessary.
|
protected void |
afterSingletonCreation(java.lang.String beanName)
Callback after singleton creation.
|
protected void |
beforeSingletonCreation(java.lang.String beanName)
Callback before singleton creation.
|
boolean |
containsSingleton(java.lang.String beanName)
Check if this registry contains a singleton instance with the given name.
|
protected void |
destroyBean(java.lang.String beanName,
DisposableBean bean)
Destroy the given bean.
|
void |
destroySingleton(java.lang.String beanName)
Destroy the given bean.
|
void |
destroySingletons() |
java.lang.String[] |
getDependenciesForBean(java.lang.String beanName)
Return the names of all beans that the specified bean depends on, if any.
|
java.lang.String[] |
getDependentBeans(java.lang.String beanName)
Return the names of all beans which depend on the specified bean, if any.
|
java.lang.Object |
getSingleton(java.lang.String beanName)
Return the (raw) singleton object registered under the given name.
|
protected java.lang.Object |
getSingleton(java.lang.String beanName,
boolean allowEarlyReference)
Return the (raw) singleton object registered under the given name.
|
java.lang.Object |
getSingleton(java.lang.String beanName,
ObjectFactory<?> singletonFactory)
Return the (raw) singleton object registered under the given name,
creating and registering a new one if none registered yet.
|
int |
getSingletonCount()
Return the number of singleton beans registered in this registry.
|
java.lang.Object |
getSingletonMutex()
Exposes the singleton mutex to subclasses and external collaborators.
|
java.lang.String[] |
getSingletonNames()
Return the names of singleton beans registered in this registry.
|
protected boolean |
hasDependentBean(java.lang.String beanName)
Determine whether a dependent bean has been registered for the given name.
|
protected boolean |
isActuallyInCreation(java.lang.String beanName) |
boolean |
isCurrentlyInCreation(java.lang.String beanName) |
protected boolean |
isDependent(java.lang.String beanName,
java.lang.String dependentBeanName)
Determine whether the specified dependent bean has been registered as
dependent on the given bean or on any of its transitive dependencies.
|
private boolean |
isDependent(java.lang.String beanName,
java.lang.String dependentBeanName,
java.util.Set<java.lang.String> alreadySeen) |
boolean |
isSingletonCurrentlyInCreation(java.lang.String beanName)
Return whether the specified singleton bean is currently in creation
(within the entire factory).
|
protected void |
onSuppressedException(java.lang.Exception ex)
Register an Exception that happened to get suppressed during the creation of a
singleton bean instance, e.g.
|
void |
registerContainedBean(java.lang.String containedBeanName,
java.lang.String containingBeanName)
Register a containment relationship between two beans,
e.g.
|
void |
registerDependentBean(java.lang.String beanName,
java.lang.String dependentBeanName)
Register a dependent bean for the given bean,
to be destroyed before the given bean is destroyed.
|
void |
registerDisposableBean(java.lang.String beanName,
DisposableBean bean)
Add the given bean to the list of disposable beans in this registry.
|
void |
registerSingleton(java.lang.String beanName,
java.lang.Object singletonObject)
Register the given existing object as singleton in the bean registry,
under the given bean name.
|
protected void |
removeSingleton(java.lang.String beanName)
Remove the bean with the given name from the singleton cache of this factory,
to be able to clean up eager registration of a singleton if creation failed.
|
void |
setCurrentlyInCreation(java.lang.String beanName,
boolean inCreation) |
allowAliasOverriding, canonicalName, checkForAliasCircle, getAliases, hasAlias, isAlias, registerAlias, removeAlias, resolveAliases
protected static final java.lang.Object NULL_OBJECT
protected final Log logger
private final java.util.Map<java.lang.String,java.lang.Object> singletonObjects
private final java.util.Map<java.lang.String,ObjectFactory<?>> singletonFactories
private final java.util.Map<java.lang.String,java.lang.Object> earlySingletonObjects
private final java.util.Set<java.lang.String> registeredSingletons
private final java.util.Set<java.lang.String> singletonsCurrentlyInCreation
private final java.util.Set<java.lang.String> inCreationCheckExclusions
private java.util.Set<java.lang.Exception> suppressedExceptions
private boolean singletonsCurrentlyInDestruction
private final java.util.Map<java.lang.String,java.lang.Object> disposableBeans
private final java.util.Map<java.lang.String,java.util.Set<java.lang.String>> containedBeanMap
private final java.util.Map<java.lang.String,java.util.Set<java.lang.String>> dependentBeanMap
private final java.util.Map<java.lang.String,java.util.Set<java.lang.String>> dependenciesForBeanMap
public void registerSingleton(java.lang.String beanName, java.lang.Object singletonObject) throws java.lang.IllegalStateException
SingletonBeanRegistry
The given instance is supposed to be fully initialized; the registry
will not perform any initialization callbacks (in particular, it won't
call InitializingBean's afterPropertiesSet
method).
The given instance will not receive any destruction callbacks
(like DisposableBean's destroy
method) either.
When running within a full BeanFactory: Register a bean definition instead of an existing instance if your bean is supposed to receive initialization and/or destruction callbacks.
Typically invoked during registry configuration, but can also be used for runtime registration of singletons. As a consequence, a registry implementation should synchronize singleton access; it will have to do this anyway if it supports a BeanFactory's lazy initialization of singletons.
registerSingleton
in interface SingletonBeanRegistry
beanName
- the name of the beansingletonObject
- the existing singleton objectjava.lang.IllegalStateException
InitializingBean.afterPropertiesSet()
,
DisposableBean.destroy()
,
BeanDefinitionRegistry.registerBeanDefinition(java.lang.String, org.springframework.beans.factory.config.BeanDefinition)
protected void addSingleton(java.lang.String beanName, java.lang.Object singletonObject)
To be called for eager registration of singletons.
beanName
- the name of the beansingletonObject
- the singleton objectprotected void addSingletonFactory(java.lang.String beanName, ObjectFactory<?> singletonFactory)
To be called for eager registration of singletons, e.g. to be able to resolve circular references.
beanName
- the name of the beansingletonFactory
- the factory for the singleton objectpublic java.lang.Object getSingleton(java.lang.String beanName)
SingletonBeanRegistry
Only checks already instantiated singletons; does not return an Object for singleton bean definitions which have not been instantiated yet.
The main purpose of this method is to access manually registered singletons
(see SingletonBeanRegistry.registerSingleton(java.lang.String, java.lang.Object)
). Can also be used to access a singleton
defined by a bean definition that already been created, in a raw fashion.
NOTE: This lookup method is not aware of FactoryBean prefixes or aliases. You need to resolve the canonical bean name first before obtaining the singleton instance.
getSingleton
in interface SingletonBeanRegistry
beanName
- the name of the bean to look fornull
if none foundConfigurableListableBeanFactory.getBeanDefinition(java.lang.String)
protected java.lang.Object getSingleton(java.lang.String beanName, boolean allowEarlyReference)
Checks already instantiated singletons and also allows for an early reference to a currently created singleton (resolving a circular reference).
beanName
- the name of the bean to look forallowEarlyReference
- whether early references should be created or notnull
if none foundpublic java.lang.Object getSingleton(java.lang.String beanName, ObjectFactory<?> singletonFactory)
beanName
- the name of the beansingletonFactory
- the ObjectFactory to lazily create the singleton
with, if necessaryprotected void onSuppressedException(java.lang.Exception ex)
ex
- the Exception to registerprotected void removeSingleton(java.lang.String beanName)
beanName
- the name of the beangetSingletonMutex()
public boolean containsSingleton(java.lang.String beanName)
SingletonBeanRegistry
Only checks already instantiated singletons; does not return true
for singleton bean definitions which have not been instantiated yet.
The main purpose of this method is to check manually registered singletons
(see SingletonBeanRegistry.registerSingleton(java.lang.String, java.lang.Object)
). Can also be used to check whether a
singleton defined by a bean definition has already been created.
To check whether a bean factory contains a bean definition with a given name,
use ListableBeanFactory's containsBeanDefinition
. Calling both
containsBeanDefinition
and containsSingleton
answers
whether a specific bean factory contains a local bean instance with the given name.
Use BeanFactory's containsBean
for general checks whether the
factory knows about a bean with a given name (whether manually registered singleton
instance or created by bean definition), also checking ancestor factories.
NOTE: This lookup method is not aware of FactoryBean prefixes or aliases. You need to resolve the canonical bean name first before checking the singleton status.
containsSingleton
in interface SingletonBeanRegistry
beanName
- the name of the bean to look forSingletonBeanRegistry.registerSingleton(java.lang.String, java.lang.Object)
,
ListableBeanFactory.containsBeanDefinition(java.lang.String)
,
BeanFactory.containsBean(java.lang.String)
public java.lang.String[] getSingletonNames()
SingletonBeanRegistry
Only checks already instantiated singletons; does not return names for singleton bean definitions which have not been instantiated yet.
The main purpose of this method is to check manually registered singletons
(see SingletonBeanRegistry.registerSingleton(java.lang.String, java.lang.Object)
). Can also be used to check which singletons
defined by a bean definition have already been created.
getSingletonNames
in interface SingletonBeanRegistry
null
)SingletonBeanRegistry.registerSingleton(java.lang.String, java.lang.Object)
,
BeanDefinitionRegistry.getBeanDefinitionNames()
,
ListableBeanFactory.getBeanDefinitionNames()
public int getSingletonCount()
SingletonBeanRegistry
Only checks already instantiated singletons; does not count singleton bean definitions which have not been instantiated yet.
The main purpose of this method is to check manually registered singletons
(see SingletonBeanRegistry.registerSingleton(java.lang.String, java.lang.Object)
). Can also be used to count the number of
singletons defined by a bean definition that have already been created.
getSingletonCount
in interface SingletonBeanRegistry
SingletonBeanRegistry.registerSingleton(java.lang.String, java.lang.Object)
,
BeanDefinitionRegistry.getBeanDefinitionCount()
,
ListableBeanFactory.getBeanDefinitionCount()
public void setCurrentlyInCreation(java.lang.String beanName, boolean inCreation)
public boolean isCurrentlyInCreation(java.lang.String beanName)
protected boolean isActuallyInCreation(java.lang.String beanName)
public boolean isSingletonCurrentlyInCreation(java.lang.String beanName)
beanName
- the name of the beanprotected void beforeSingletonCreation(java.lang.String beanName)
The default implementation register the singleton as currently in creation.
beanName
- the name of the singleton about to be createdisSingletonCurrentlyInCreation(java.lang.String)
protected void afterSingletonCreation(java.lang.String beanName)
The default implementation marks the singleton as not in creation anymore.
beanName
- the name of the singleton that has been createdisSingletonCurrentlyInCreation(java.lang.String)
public void registerDisposableBean(java.lang.String beanName, DisposableBean bean)
Disposable beans usually correspond to registered singletons, matching the bean name but potentially being a different instance (for example, a DisposableBean adapter for a singleton that does not naturally implement Spring's DisposableBean interface).
beanName
- the name of the beanbean
- the bean instancepublic void registerContainedBean(java.lang.String containedBeanName, java.lang.String containingBeanName)
Also registers the containing bean as dependent on the contained bean in terms of destruction order.
containedBeanName
- the name of the contained (inner) beancontainingBeanName
- the name of the containing (outer) beanregisterDependentBean(java.lang.String, java.lang.String)
public void registerDependentBean(java.lang.String beanName, java.lang.String dependentBeanName)
beanName
- the name of the beandependentBeanName
- the name of the dependent beanprotected boolean isDependent(java.lang.String beanName, java.lang.String dependentBeanName)
beanName
- the name of the bean to checkdependentBeanName
- the name of the dependent beanprivate boolean isDependent(java.lang.String beanName, java.lang.String dependentBeanName, java.util.Set<java.lang.String> alreadySeen)
protected boolean hasDependentBean(java.lang.String beanName)
beanName
- the name of the bean to checkpublic java.lang.String[] getDependentBeans(java.lang.String beanName)
beanName
- the name of the beanpublic java.lang.String[] getDependenciesForBean(java.lang.String beanName)
beanName
- the name of the beanpublic void destroySingletons()
public void destroySingleton(java.lang.String beanName)
destroyBean
if a corresponding disposable bean instance is found.beanName
- the name of the beandestroyBean(java.lang.String, org.springframework.beans.factory.DisposableBean)
protected void destroyBean(java.lang.String beanName, DisposableBean bean)
beanName
- the name of the beanbean
- the bean instance to destroypublic final java.lang.Object getSingletonMutex()
Subclasses should synchronize on the given Object if they perform any sort of extended singleton creation phase. In particular, subclasses should not have their own mutexes involved in singleton creation, to avoid the potential for deadlocks in lazy-init situations.
getSingletonMutex
in interface SingletonBeanRegistry
null
)