Неоднозначное отображение найдено с ControllerClassNameHandlerMapping


Как сопоставить запросы к методам без явных аннотаций поверх методов? Например, следующий запрос:

http://somedomain:8080/SampleSpring/access/loginFailed

Должно быть сопоставлено с

" loginFailed "метод" AccessController " без необходимости явной аннотации на метод, как:

@RequestMapping("/access/loginFailed")

Вот моя весенняя конфигурация:

<context:component-scan base-package="com.robikcodes.samplespring"/>
<mvc:annotation-driven/>

<bean class="org.springframework.web.servlet.mvc.support.ControllerClassNameHandlerMapping">
        <property name="basePackage" value="com.robikcodes.samplespring.controller"/>
        <property name="caseSensitive" value="true"/>
        <property name="defaultHandler">
            <bean class="org.springframework.web.servlet.mvc.UrlFilenameViewController"/>
        </property>
</bean>   

<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
            <property name="prefix" value="/WEB-INF/views/"/>
            <property name="suffix" value=".jsp"/>
</bean>

Вот мой контроллер:

@Controller
public class AccessController{

    @RequestMapping(method = RequestMethod.GET)
    public void login(ModelMap m) {}

    @RequestMapping(method = RequestMethod.GET)
    public String loginFailed(ModelMap m) {
        m.addAttribute("error", "true");
        return "access/login";
    }

    @RequestMapping(method = RequestMethod.GET)
    public String logout(ModelMap m) {
        m.addAttribute("logoutStatus","true");
        return "access/login";
    }
}

Я получил следующую ошибку (похоже, что только метод входа был отображен правильно):

org.apache.catalina.LifecycleException: org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping#0': Invocation of init method failed; nested exception is java.lang.IllegalStateException: Ambiguous mapping found. Cannot map 'accessController' bean method 
public java.lang.String com.robikcodes.samplespring.controller.AccessController.logout(org.springframework.ui.ModelMap)
to {[],methods=[GET],params=[],headers=[],consumes=[],produces=[],custom=[]}: There is already 'accessController' bean method
public void com.robikcodes.samplespring.controller.AccessController.login(org.springframework.ui.ModelMap) mapped.
1 3

1 ответ:

Вы используете ControllerClassNameHandlerMapping с предположением, которое не является правильным; из Java doc:

Реализация HandlerMapping, которая следует простой конвенции для создание сопоставлений URL-адресов из имен классов зарегистрированных Бобы контроллера, а также аннотированные бобы @Controller.

В документации не говорится, что он также следуетименам методов . Основной ссылкой сравнения "отображений обработчиков" для вашего контроллера является @RequestMapping аннотации ставьте на свои методы. Итак, с помощью вашего контроллера Spring считывает их как:

{methods=[GET],params=[],headers=[],consumes=[],produces=[],custom=[]}

Для всех определенных методов в AccessController, который имеет следующий @RequestMapping:

@RequestMapping(method = RequestMethod.GET)
Вот почему вы видите неоднозначное исключение.

Насколько я понимаю, наиболее чистым решением является использование атрибута value @RequestMapping для определения различных URI запросов. На самом деле не рекомендуется искать решение, которое пытается сопоставить URI запроса с именами методов.