我正在为我的应用程序开发RESTful API。所有getter(使用HTTP GET)都可以正常工作。我无法使用save方法(使用POST)来工作。
我正在使用HTML表单和RESTClient进行测试。
这是我的控制器
@Controller
public class EntitiesController {
@RequestMapping(value="/ci/save/", method = RequestMethod.POST)
public ModelAndView saveConfigurationItem(@RequestBody ConfigurationItem body) {
System.out.println("saveConfigurationItem: body=" + body);
return createModelAndView("ci", Collections.emptyList());
}
}
当客户端发布ConfigurationItem时,预计会调用此方法。 我正在使用自定义序列化格式。它不是XML或JSON。它是VCard或VCalendar格式。对于我的第一次测试,我使用了以下VCard:
BEGIN:VCARD
N:Pooh;Winnie
FN:Winnie the Pooh
TEL:tel:+441234567
END:VCARD
我已将其发布到网址http://localhost:8080/core.solution-1.0/data/ci/save/
。
以下是我得到的回复:
415
The server refused this request because the request entity is in a format not
supported by the requested resource for the requested method ()
(*)ConfigurationItem是一个抽象类。 CardEntry扩展了它。我试过了。
我尝试将方法参数更改为String
。在这种情况下,调用该方法但字符串为空。在遵循我在Web中看到的建议之一时,同样的情况发生了我将参数类型更改为MultiValueMap并从简单的HTML表单发送请求。
我看到marshal()根本没有被调用。
怎么了?
这就是我所拥有的。 (我只在这里提供相关代码。)
春季配置
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:oxm="http://www.springframework.org/schema/oxm"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd
http://www.springframework.org/schema/oxm http://www.springframework.org/schema/oxm/spring-oxm-3.0.xsd">
<import resource="classes/spring-config-prod.xml"/>
<context:component-scan base-package="com.mycompany.solution.service" />
<bean class="org.springframework.web.servlet.view.BeanNameViewResolver" />
<bean id="ciCardView" class="com.mycompany.solution.service.VFormatView">
<constructor-arg>
<bean class="com.mycompany.solution.service.VFormatMarshaller">
<property name="packagesToScan" value="com.mycompany.solution.entity"/>
</bean>
</constructor-arg>
</bean>
</beans>
的Marshaller
public class VFormatMarshaller implements Marshaller, Unmarshaller {
@Override
public void marshal(Object obj, Result result)
throws IOException/*, XmlMappingException*/ {
System.out.println("VFormatMarshaller.marshal(" + obj + ")");
marshalStreamResult(obj, (StreamResult)result);
}
@Override
public boolean supports(Class<?> paramClass) {
System.out.println("VFormatMarshaller.supports(" + paramClass + ")");
boolean supports = new HashSet<String>(Arrays.asList(packagesToScan)).contains(paramClass.getPackage().getName());
if (supports) {
return supports;
}
return Collection.class.isAssignableFrom(paramClass);
}
@Override
public Object unmarshal(Source source) throws IOException/*, XmlMappingException*/ {
System.out.println("VFormatMarshaller.unmarshal(" + source + ")");
return unmarshalStreamSource((StreamSource)source);
}
//// .............................
}
查看(这只是为了覆盖内容类型)
public class VFormatView extends MarshallingView {
public VFormatView() {
super();
setContentType("application/vcard");
System.out.println("VFormatView()");
}
public VFormatView(Marshaller marshaller) {
super(marshaller);
setContentType("application/vcard");
System.out.println("VFormatView(" + marshaller + ")");
}
}
答案 0 :(得分:2)
@RequestBody
/ @ResponseBody
的层次结构支持HttpMessageConverter
/ ViewResolver
,与HttpMessageConverter
s完全不同。
在这种情况下,您需要使用适当的marshaller / unmarshaller和内容类型配置MarshallingHttpMessageConverter
(如果您不需要依赖marshaller / unmarshaller的现有实现,则需要创建自己的HttpMessageConveter
),并将配置的实例提供给AnnotationMethodHandlerAdapter
。
配置自定义BeanPostProcessor
的最不具侵入性的方法是创建public class Configurer implements BeanPostProcessor {
public void postProcessAfterInitialization(String name, Object bean) {
if (bean instanceof AnnotationMethodHandlerAdapter) {
AnnotationMethodHandlerAdapter a = (AnnotationMethodHandlerAdapter) bean;
HttpMessageConverter[] convs = a.getMessageConverters();
... add new converter ...
a.setMessageConverters(convs);
}
}
...
}
,如下所示:
{{1}}