Spring-MVC 406 不可接受,而不是 JSON 响应

2022-09-04 07:39:34

我正在尝试使用Spring 3.0.6返回JSON响应,但我得到406响应“不可接受”,其描述是:“此请求标识的资源只能根据请求”接受“标头()生成具有不可接受的特征的响应。

我知道以前有人问过一个非常相似的问题,但是我无法让它为我的项目工作,尽管我做了很多测试,我不明白我做错了什么。

在我的Maven pom中.xml我有以下内容:

<dependency>
  <groupId>org.codehaus.jackson</groupId>
  <artifactId>jackson-mapper-asl</artifactId>
  <version>1.8.5</version>
  <scope>compile</scope>
</dependency>
<dependency>
  <groupId>org.codehaus.jackson</groupId>
  <artifactId>jackson-core-asl</artifactId>
  <version>1.8.5</version>
  <scope>compile</scope>
</dependency>

在web.xml我引用了webmvc-config.xml,日志确认已加载。

<servlet>
    <servlet-name>mainServlet</servlet-name>
    <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
    <init-param>
        <param-name>contextConfigLocation</param-name>
        <param-value>/WEB-INF/spring/webmvc-config.xml</param-value>
    </init-param>
    <load-on-startup>1</load-on-startup>
</servlet>

在webmvc-config中.xml我有以下内容:

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

我的控制器是:

@Controller
public class ClassifiedController {

    @RequestMapping(value = "/classified/{idClassified}", headers = "Accept=*/*",
                    method = RequestMethod.GET)
    @ResponseBody
    public final Classified getClassified(@PathVariable final int idClassified) {
        ...

我尝试使用或不使用标头参数具有相同的结果。如果我直接用Firefox调用URL,请求标头包含以下内容(使用firebug检查):

Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8

如果我使用以下JQuery:

$.ajax({
        url: '/classified/38001',
        type: 'GET',
        dataType: 'json'
});

发送以下标头:

Accept: application/json, text/javascript, */*; q=0.01

在这两种情况下,结果都是 406 错误。我不知道我还应该检查什么才能使其正常工作。


更新:我决定通过Spring进行调试,我发现Jackson被正确调用,并且在org.codehaus.jackson.map.ser.StdSerializer中,_findExplicitUntypedSerializer方法包含以下代码:

try {
    return _createAndCacheUntypedSerializer(runtimeType, property);
} catch (Exception e) {
    return null;
}

这是不幸的,因为隐藏了问题的根源。使用调试器,我发现该异常包含一个非常具有描述性的错误消息:

Conflicting getter definitions for property "reminded": 
ClassifiedImpl#isReminded(0 params) vs
ClassifiedImpl#getReminded(0 params)

现在我看到错误消息是一个愚蠢的错误,很容易修复,但没有它,它就不那么明显了。实际上,修复了这个问题,导致了一个有效的序列化。


答案 1

在 DispatcherServlet-servlet 中添加以下内容.xml。

<bean id="jacksonMessageConverter" class="org.springframework.http.converter.json.MappingJacksonHttpMessageConverter"></bean>
<bean class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter">
    <property name="messageConverters">
        <list>
            <ref bean="jacksonMessageConverter"/>
        </list>
    </property>
</bean>

答案 2

我偶然发现了与Spring MVC相同的错误(406:内容不可接受),@RestController注释。

弹簧处理程序:

@RequestMapping(value = "/stuff-acknowledgment/{id}", produces ="application/json;charset=UTF-8", headers="Accept=*")
public Message acknowledgeStuff(@PathVariable("id") String id, @ModelAttribute("ack") AckBean acquittement) {

观察:

  • URI 的格式为:http://www.host.com/stuff-acknowledgment/{id}
  • 但是有一个非常特殊的格式:(或者你能想到的任何扩展名)。$idxxxcomplicatedhashxxx.png

因此:

Spring MVC解释扩展并希望产生相同mime类型的结果(即使我将其定义为路径变量),这里是MIME类型,即使我告诉他生成JSON。因此,将引发 406 异常。"image/png"

修复:

删除URI中的扩展名,或删除PathVariable并将其放在正文中,或者在pathVariable后面添加一个后缀(未经测试,但也应该工作),重点是避免在URI的末尾使用文件扩展名。".png"

P.S.:我知道它没有回答问题中的具体问题(在更新中提供了解决方案),但我在搜索该问题时发现了SO线程,并在此处发布我的修复程序以供记录,希望它可以帮助将来的某人。