这个方法是判断给定的 name 和 alias 之间是否具备关联关系。判断的逻辑就是先去 aliasMap 中,根据 alias 查出来这个 alias 所对应的真实 beanName,即 registeredName,然后判断 registeredName 和 name 是否相等,如果相等就直接返回,如果不相等就继续递归调用,为什么要递归呢?因为 aliasMap 中存在的别名可能是这样的:
a->b
b->c
c->d
即 a 是 b 的别名,b 是 c 的别名,c 是 d 的别名,现在如果想要判断 a 和 d 之间的关系,那么根据 a 查出来的 b 显然不等于 d,所以要继续递归,再根据 b 查 c,根据 c 查到 d,这样就能确定 a 和 d 是否有关系了。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
@Override public String[] getAliases(String name) { List<String> result = new ArrayList<>(); synchronized (this.aliasMap) { retrieveAliases(name, result); } return StringUtils.toStringArray(result); } privatevoidretrieveAliases(String name, List<String> result){ this.aliasMap.forEach((alias, registeredName) -> { if (registeredName.equals(name)) { result.add(alias); retrieveAliases(alias, result); } }); }
getAliases 方法是根据传入的 name 找到其对应的别名,但是由于别名可能存在多个,所以调用 retrieveAliases 方法递归去查找所有的别名,将找到的别名都存入到一个集合中,最终将集合转为数组返回。
1 2 3 4 5 6 7
protectedvoidcheckForAliasCircle(String name, String alias){ if (hasAlias(alias, name)) { thrownew IllegalStateException("Cannot register alias '" + alias + "' for name '" + name + "': Circular reference - '" + name + "' is a direct or indirect alias for '" + alias + "' already"); } }
这个方法用来检查别名是否存在死结,即 a 是 b 的别名,b 是 a 的别名这种情况。检查的方式很简单,就是调用 hasAlias 方法,但是将传入的两个参数颠倒过来就可以了。
大家看到,首先这里将 aliasMap 复制一份,生成一个 aliasCopy,然后进行遍历。在遍历时,根据 valueResolver 将引用使用的占位符解析为真正的字符,如果解析出来的。如果解析出来的 name 和别名是相同的,那么显然是有问题的,就需要把这个别名移除掉。
继续判断,如果解析出来的别名和原本的别名不相等(说明别名使用了占位符),那么就去检查一下这个别名对应的 name,如果这个 name 已经存在,且等于占位符解析出来的 name,说明这个别名已经被定义过了,即重复定义,那么就把别名移除掉即可。如果这个别名指向的 name 和占位符解析出来的 name 不相等,说明试图让一个别名指向两个 bean,那么就直接抛出异常了。
如果解析出来的别名还没有指向 name 属性的话,那么就正常处理,检查是否存在死结、移除带占位符的别名,存入解析之后的别名。
protectedvoidprocessAliasRegistration(Element ele){ String name = ele.getAttribute(NAME_ATTRIBUTE); String alias = ele.getAttribute(ALIAS_ATTRIBUTE); boolean valid = true; if (!StringUtils.hasText(name)) { getReaderContext().error("Name must not be empty", ele); valid = false; } if (!StringUtils.hasText(alias)) { getReaderContext().error("Alias must not be empty", ele); valid = false; } if (valid) { try { getReaderContext().getRegistry().registerAlias(name, alias); } catch (Exception ex) { getReaderContext().error("Failed to register alias '" + alias + "' for bean with name '" + name + "'", ele, ex); } getReaderContext().fireAliasRegistered(name, alias, extractSource(ele)); } }
可以看到,这里也是从 XML 文件中的别名标签上,提取出来 name 和 alias 属性值,最后调用 registerAlias 方法进行注册。