这一次我们来看一下HandlerMapping的另一种实现方式BeanNameUrlHandlerMapping。
配置文件
<bean name="/hello.htm" class="com.raistudies.ui.comtroller.HelloController"/>
<bean name="/sayHello*" class="com.raistudies.ui.comtroller.HelloController"/>
<bean id="urlHandler" class="org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping"/>
好像完全看不出来"/hello.htm"和"/sayHello*"使用了BeanNameUrlHandlerMapping,因为它们之间直接都是分离的,但为什么这样可以调用BeanNameUrlHandlerMapping,我们看源码来理解。
BeanNameUrlHandlerMapping类
public class BeanNameUrlHandlerMapping extends AbstractDetectingUrlHandlerMapping {
/**
* Checks name and aliases of the given bean for URLs, starting with "/".
*/
@Override
protected String[] determineUrlsForHandler(String beanName) {
List<String> urls = new ArrayList<>();
if (beanName.startsWith("/")) {
urls.add(beanName);
}
String[] aliases = obtainApplicationContext().getAliases(beanName);
for (String alias : aliases) {
if (alias.startsWith("/")) {
urls.add(alias);
}
}
return StringUtils.toStringArray(urls);
}
}
BeanNameUrlHandlerMapping类中只有一个determineUrlsForHandler,这个方法很短,所做的功能就是找到beanName所对应的urls,简单点理解urls应该就是本名+别名。
不过有个细节我们需要注意,那就是BeanNameUrlHandlerMapping只会处理那些前缀为"/"的urls。
接下来看BeanNameUrlHandlerMapping的父类。
AbstractDetectingUrlHandlerMapping类
首先从initApplicationContext()看起
public void initApplicationContext() throws ApplicationContextException {
super.initApplicationContext();
detectHandlers();
}
里面调用了detectHandlers()
protected void detectHandlers() throws BeansException {
ApplicationContext applicationContext = obtainApplicationContext();
if (logger.isDebugEnabled()) {
logger.debug("Looking for URL mappings in application context: " + applicationContext);
}
String[] beanNames = (this.detectHandlersInAncestorContexts ?
BeanFactoryUtils.beanNamesForTypeIncludingAncestors(applicationContext, Object.class) :
applicationContext.getBeanNamesForType(Object.class));
// Take any bean name that we can determine URLs for.
for (String beanName : beanNames) {
String[] urls = determineUrlsForHandler(beanName);
if (!ObjectUtils.isEmpty(urls)) {
// URL paths found: Let's consider it a handler.
registerHandler(urls, beanName);
}
else {
if (logger.isDebugEnabled()) {
logger.debug("Rejected bean name '" + beanName + "': no URL paths identified");
}
}
}
}
看到下面这行代码
String[] beanNames = (this.detectHandlersInAncestorContexts ?
BeanFactoryUtils.beanNamesForTypeIncludingAncestors(applicationContext, Object.class) :
applicationContext.getBeanNamesForType(Object.class));
我们发现原来AbstractDetectingUrlHandlerMapping将配置文件中所有的bean都扫描了出来,因为Java中所有类都继承Object。这也就解释了最初的配置文件部分的疑问。
那么剩下的逻辑就很简单了,就是将所有的以"/"开头的url与其对应handler进行绑定。
至于绑定逻辑和上一篇文章中所讲的SimpleUrlHandlerMapping一样了,因为它们都继承了同一个父类AbstractUrlHandlerMapping。
总结
从源码中我们可以理解BeanNameUrlHandlerMapping作为默认的HandlerMapping的原因,因为BeanNameUrlHandlerMapping扫描了配置文件中所有的bean。
注意事项:只有name开头为"/"的bean才会被最终映射。