我想使用Spring MVC发布包含一些JSON数据的文件。所以我开发了一个休息服务
@RequestMapping(value = "/servicegenerator/wsdl", method = RequestMethod.POST,consumes = { "multipart/mixed", "multipart/form-data" })
@ResponseBody
public String generateWSDLService(@RequestPart("meta-data") WSDLInfo wsdlInfo,@RequestPart("file") MultipartFile file) throws WSDLException, IOException,
JAXBException, ParserConfigurationException, SAXException, TransformerException {
return handleWSDL(wsdlInfo,file);
}
当我从其他客户端发送请求时
content-Type = multipart/form-data or multipart/mixed
,我得到了下一个例外:
org.springframework.web.multipart.support.MissingServletRequestPartException
有人可以帮我解决这个问题吗?
我可以使用@RequestPart
将Multipart和JSON发送到服务器吗?
答案 0 :(得分:142)
这就是我用JSON数据实现Spring MVC Multipart Request的方法。
基于Spring 4.0.2 Release中的RESTful服务,可以使用@RequestPart实现HTTP请求,第一部分为XML或JSON格式的数据,第二部分为文件。以下是示例实现。
Controller中的Rest服务将混合使用@RequestPart和MultipartFile来提供此类Multipart + JSON请求。
@RequestMapping(value = "/executesampleservice", method = RequestMethod.POST,
consumes = {"multipart/form-data"})
@ResponseBody
public boolean executeSampleService(
@RequestPart("properties") @Valid ConnectionProperties properties,
@RequestPart("file") @Valid @NotNull @NotBlank MultipartFile file) {
return projectService.executeSampleService(properties, file);
}
创建FormData对象。
使用以下步骤之一将文件附加到FormData对象。
formData.append("file", document.forms[formName].file.files[0]);
formData.append("file", myFile, "myfile.txt");
或formData.append("file", myBob, "myfile.txt");
使用字符串化的JSON数据创建一个blob,并将其附加到FormData对象。这导致多部分请求中第二部分的Content-type为" application / json"而不是文件类型。
将请求发送到服务器。
要求明细:
Content-Type: undefined
。这会导致浏览器将Content-Type设置为multipart / form-data并正确填充边界。手动将Content-Type设置为multipart / form-data将无法填写请求的边界参数。
formData = new FormData();
formData.append("file", document.forms[formName].file.files[0]);
formData.append('properties', new Blob([JSON.stringify({
"name": "root",
"password": "root"
})], {
type: "application/json"
}));
method: "POST",
headers: {
"Content-Type": undefined
},
data: formData
Accept:application/json, text/plain, */*
Content-Type:multipart/form-data; boundary=----WebKitFormBoundaryEBoJzS3HQ4PgE1QB
------WebKitFormBoundaryvijcWI2ZrZQ8xEBN
Content-Disposition: form-data; name="file"; filename="myfile.txt"
Content-Type: application/txt
------WebKitFormBoundaryvijcWI2ZrZQ8xEBN
Content-Disposition: form-data; name="properties"; filename="blob"
Content-Type: application/json
------WebKitFormBoundaryvijcWI2ZrZQ8xEBN--
答案 1 :(得分:8)
这必须有效!
客户(角度):
$scope.saveForm = function () {
var formData = new FormData();
var file = $scope.myFile;
var json = $scope.myJson;
formData.append("file", file);
formData.append("ad",JSON.stringify(json));//important: convert to JSON!
var req = {
url: '/upload',
method: 'POST',
headers: {'Content-Type': undefined},
data: formData,
transformRequest: function (data, headersGetterFunction) {
return data;
}
};
Backend-Spring Boot:
@RequestMapping(value = "/upload", method = RequestMethod.POST)
public @ResponseBody
Advertisement storeAd(@RequestPart("ad") String adString, @RequestPart("file") MultipartFile file) throws IOException {
Advertisement jsonAd = new ObjectMapper().readValue(adString, Advertisement.class);
//do whatever you want with your file and jsonAd
答案 2 :(得分:2)
正如文件所说:
当“multipart / form-data”请求的一部分被识别时引发 它的名字无法找到。
这可能是因为请求不是multipart / form-data 因为该部分不在请求中,或者因为Web 未正确配置应用程序以处理multipart 请求 - 例如没有MultipartResolver。
答案 3 :(得分:0)
我们已经在项目中看到,带有JSON和文件的发布请求在前端和后端开发人员之间造成了很多混乱,导致不必要的时间浪费。
这是一种更好的方法:将文件字节数组转换为Base64字符串并以JSON发送。
public Class UserDTO {
private String firstName;
private String lastName;
private FileDTO profilePic;
}
public class FileDTO {
private String base64;
// just base64 string is enough. If you want, send additional details
private String name;
private String type;
private String lastModified;
}
@PostMapping("/user")
public String saveUser(@RequestBody UserDTO user) {
byte[] fileBytes = Base64Utils.decodeFromString(user.getProfilePic().getBase64());
....
}
用于将文件转换为base64字符串的JS代码:
var reader = new FileReader();
reader.readAsDataURL(file);
reader.onload = function () {
const userDTO = {
firstName: "John",
lastName: "Wick",
profilePic: {
base64: reader.result,
name: file.name,
lastModified: file.lastModified,
type: file.type
}
}
// post userDTO
};
reader.onerror = function (error) {
console.log('Error: ', error);
};
答案 4 :(得分:0)
针对 Angular2+ 用户。尝试在混合部分请求中发送 JSON 负载,如下所示。
formData.append("jsonPayload", new Blob([JSON.stringify(json)], {
type: "application/json"
}));
下面给出了完整的功能。
submit() {
const formData = new FormData();
formData.append('file', this.myForm.get('fileSource').value);
var json = {
"key":"value"
};
formData.append("jsonPayload", new Blob([JSON.stringify(json)], {
type: "application/json"
}));
this.http.post('http://localhost:8080/api/mixed-part-endpoint', formData)
.subscribe(res => {
console.log(res);
alert('Uploaded Successfully.');
})
}