Spring源碼之路-DefaulListableBeanFactory
SimpleAliasRegistry接口總結
- 此類是AliasRegistry 的簡單實現,一個名字可以有很多別名,但內部只維護着一個ConcurrentHashMap,並且泛型也是String String。仔細看過源碼以後知道,他是將別名作爲key,真實名字作爲value這種方式存儲別名和真實名稱之間的關係(這種方式說起來比較簡單,但是實現起來還是有點複雜的,會有點饒,可以看下源碼實現)
- 類中多了個方法
public void resolveAliases(StringValueResolver valueResolver)
這個方法不在接口的方法裏面,所以是必須依賴當前的實現類,而不是依賴接口。
源碼註釋的大概翻譯
/**
* {@link AliasRegistry}接口的簡單實現。
* 用作{@link org.springframework.beans.factory.support.BeanDefinitionRegistry}
* 實現的基類
*
* @author Juergen Hoeller
* @author Qimiao Chen
* @since 2.5.2
*/
public class SimpleAliasRegistry implements AliasRegistry {
/** Logger available to subclasses. */
protected final Log logger = LogFactory.getLog(getClass());
/ **從別名映射到規範名稱。 * /
private final Map<String, String> aliasMap = new ConcurrentHashMap<>(16);
@Override
public void registerAlias(String name, String alias) {
Assert.hasText(name, "'name' must not be empty");
Assert.hasText(alias, "'alias' must not be empty");
synchronized (this.aliasMap) {
if (alias.equals(name)) {
this.aliasMap.remove(alias);
if (logger.isDebugEnabled()) {
logger.debug("Alias definition '" + alias + "' ignored since it points to same name");
}
}
else {
String registeredName = this.aliasMap.get(alias);
if (registeredName != null) {
if (registeredName.equals(name)) {
// An existing alias - no need to re-register
return;
}
if (!allowAliasOverriding()) {
throw new IllegalStateException("Cannot define alias '" + alias + "' for name '" +
name + "': It is already registered for name '" + registeredName + "'.");
}
if (logger.isDebugEnabled()) {
logger.debug("Overriding alias '" + alias + "' definition for registered name '" +
registeredName + "' with new target name '" + name + "'");
}
}
checkForAliasCircle(name, alias);
this.aliasMap.put(alias, name);
if (logger.isTraceEnabled()) {
logger.trace("Alias definition '" + alias + "' registered for name '" + name + "'");
}
}
}
}
/ **
* 確定是否允許別名替代。
* 默認爲{@code true}。
* /
protected boolean allowAliasOverriding() {
return true;
}
/**
* 確定給定名稱是否已註冊給定別名。
* @param name 要檢查的名稱
* @param alias 要查找的別名
* @since 4.2.1
*/
public boolean hasAlias(String name, String alias) {
String registeredName = this.aliasMap.get(alias);
return ObjectUtils.nullSafeEquals(registeredName, name) || (registeredName != null
&& hasAlias(name, registeredName));
}
@Override
public void removeAlias(String alias) {
synchronized (this.aliasMap) {
String name = this.aliasMap.remove(alias);
if (name == null) {
throw new IllegalStateException("No alias '" + alias + "' registered");
}
}
}
@Override
public boolean isAlias(String name) {
return this.aliasMap.containsKey(name);
}
@Override
public String[] getAliases(String name) {
List<String> result = new ArrayList<>();
synchronized (this.aliasMap) {
retrieveAliases(name, result);
}
return StringUtils.toStringArray(result);
}
/**
* 以傳遞方式檢索給定名稱的所有別名。
* @param name 目標名稱,用以查找別名
* @param result 生成的別名列表
*/
private void retrieveAliases(String name, List<String> result) {
this.aliasMap.forEach((alias, registeredName) -> {
if (registeredName.equals(name)) {
result.add(alias);
retrieveAliases(alias, result);
}
});
}
/**
* 解析所有在此註冊表中註冊的別名目標名稱和別名,將給定的
* {@link StringValueResolver}應用於它們。
* 例如,值解析器可以解析目標bean名稱甚至別名中的佔位符
* @param valueResolver 要應用的StringValueResolver
*/
public void resolveAliases(StringValueResolver valueResolver) {
Assert.notNull(valueResolver, "StringValueResolver must not be null");
synchronized (this.aliasMap) {
Map<String, String> aliasCopy = new HashMap<>(this.aliasMap);
aliasCopy.forEach((alias, registeredName) -> {
String resolvedAlias = valueResolver.resolveStringValue(alias);
String resolvedName = valueResolver.resolveStringValue(registeredName);
if (resolvedAlias == null || resolvedName == null || resolvedAlias.equals(resolvedName)) {
this.aliasMap.remove(alias);
}
else if (!resolvedAlias.equals(alias)) {
String existingName = this.aliasMap.get(resolvedAlias);
if (existingName != null) {
if (existingName.equals(resolvedName)) {
// Pointing to existing alias - just remove placeholder
this.aliasMap.remove(alias);
return;
}
throw new IllegalStateException(
"Cannot register resolved alias '" + resolvedAlias + "' (original: '" + alias +
"') for name '" + resolvedName + "': It is already registered for name '" +
registeredName + "'.");
}
checkForAliasCircle(resolvedName, resolvedAlias);
this.aliasMap.remove(alias);
this.aliasMap.put(resolvedAlias, resolvedName);
}
else if (!registeredName.equals(resolvedName)) {
this.aliasMap.put(alias, resolvedName);
}
});
}
}
/**
* 檢查給定名稱是否已經在另一個方向上以別名的形式指向給定別名,
* 並預先捕獲了循環引用並引發了相應的IllegalStateException。
* in the other direction already, catching a circular reference upfront
* and throwing a corresponding IllegalStateException.
* @param name 候選名稱
* @param alias 候選別名
* @see #registerAlias
* @see #hasAlias
*/
protected void checkForAliasCircle(String name, String alias) {
if (hasAlias(alias, name)) {
throw new IllegalStateException("Cannot register alias '" + alias +
"' for name '" + name + "': Circular reference - '" +
name + "' is a direct or indirect alias for '" + alias + "' already");
}
}
/**
* 確定原始名稱,將別名解析爲規範名稱。
* @param name 用戶指定的名稱
* @return 轉換後的名稱
*/
public String canonicalName(String name) {
String canonicalName = name;
// Handle aliasing...
String resolvedName;
do {
resolvedName = this.aliasMap.get(canonicalName);
if (resolvedName != null) {
canonicalName = resolvedName;
}
}
while (resolvedName != null);
return canonicalName;
}
}