我有以下方法:
@RequestMapping(value = "/app/write", method = RequestMethod.POST, consumes = "application/json", produces = "application/json")
public
@ResponseBody
Status writeBuildData(@RequestBody JsonObject templateDataJSON){}
基本上,我发出一个 Ajax
POST
请求发送 JSON,结果我总是得到一个空的 JsonObject {}
JsonObject templateDataJSON = "{}";
但是如果我使用 String 而不是
JsonObject
,我会得到正确的值。
这个应用程序是用Spring Mvc 4.1.4制作的。
依赖关系:
compile 'org.codehaus.jackson:jackson-mapper-asl:1.9.13'
compile 'com.google.code.gson:gson:2.3.1'
知道我缺少什么以及为什么
JsonObject
没有被注射并且总是给我{}
吗?
Spring 不再支持 Jackson 1 作为消息转换器实现。
所以你的
compile 'org.codehaus.jackson:jackson-mapper-asl:1.9.13'
实际上对于 Spring 来说是没有意义的。
你的
compile 'com.google.code.gson:gson:2.3.1'
将导致Spring使用
GsonHttpMessageConverter
,并且基本上,做
String json = "{\"random\":\"42\"}";
Gson gson = new Gson();
JsonObject jsonObject = gson.fromJson(json, JsonObject.class);
JsonObject
是Gson类型。 Gson 意识到了这一点,并且知道如何将 JSON 对象 json 反序列化到其中。这将正常工作并生成一个值为 的
JsonObject
{"random":"42"}
既然你说你得到的是空的
JsonObject
,我只能假设你的类路径上有 Jackson 2。
Spring 在
HttpMessageConverter
之前注册 Jackson MappingJackson2HttpMessageConverter
、GsonHttpMessageConverter
(如果两者都存在于类路径中)。
使用 Jackson,Spring 基本上会反序列化你的请求主体
ObjectMapper mapper = new ObjectMapper();
mapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
JsonObject jsonObject = mapper.readValue(json, JsonObject.class);
您将在
中记下结果{}
这是因为 Jackson 对类型
JsonObject
一无所知,因此它必须动态构建反序列化策略。该策略取决于 Jackson 定义为 setter 的属性(用于反序列化的上下文)或任何用 @JsonProperty
注释的属性,而 JsonObject
显然没有。所以它基本上认为类型 JsonObject
没有任何属性(或者可能没有出现在您的自定义 JSON 内容中)。因此,因为它忽略任何未知属性(这会导致它抛出异常),所以它只是返回一个新的空 JsonObject
对象。
一种解决方案是从类路径中删除 Jackson 2。另一个解决方案是按照您想要的顺序显式添加
HttpMessageConverter
实例。
Sotirios Delimanolis 的回答完美地解释了它失败的原因。
但是有一个解决方法:
@RequestBody Map<String, String> json
这样您就可以继续使用 Jackson HttpMessageConverter 并使用有效负载中的自定义对象。
其实很简单
@PostMapping
public ResponseEntity<?> yourMethod(@RequestBody String data) {
JSONObject jsonObject = new JSONObject(data);
//use the jsonObject now
}
JsonObject 代表什么?您应该使用代表您发送的 json 的对象。类似于
public class Foo {
String foo;
String bar;
}
改为 JsonObject
和 json 类似:
{
"foo" : "val",
"bar" : "val"
}
对我来说,只需在我的项目 application.properties 上添加“spring.http.converters.preferred-json-mapper=gson”并实现 WebMvcConfigurer 即可,如下所示...
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.converter.HttpMessageConverter;
import org.springframework.http.converter.json.GsonHttpMessageConverter;
import org.springframework.web.servlet.config.annotation.EnableWebMvc;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;
import com.google.gson.Gson;
import com.google.gson.GsonBuilder;
@Configuration
@EnableWebMvc
public class WebMvcConfig implements WebMvcConfigurer {
@Bean
public Gson gson() {
GsonBuilder gsonBuilder = new GsonBuilder()
.setDateFormat("yyyy-MM-dd HH:mm:ss")
.serializeNulls();
return gsonBuilder.create();
}
@Override
public void configureMessageConverters(List<HttpMessageConverter<?>> converters) {
GsonHttpMessageConverter gsonHttpMessageConverter = new GsonHttpMessageConverter();
gsonHttpMessageConverter.setGson(gson());
converters.add(gsonHttpMessageConverter);
}
}
现在我可以直接将RequestBody转换为JsonOject了:
@PostMapping(consumes=MediaType.APPLICATION_JSON_VALUE)
@ResponseStatus(HttpStatus.CREATED)
public void add(@RequestBody com.google.gson.JsonObject jsonObject) {
logger.info(gson.toJson(jsonObject));
}