我目前有一个采用MultipartFile的Spring MVC控制器
@RequestMapping(method = RequestMethod.POST)
public String doUpload(@RequestParam("file") final MultipartFile file) {
/* ... */
}
该文件包含将使用的csv数据,每行一个,以创建域对象列表。这很有效。
我为线数据编写了一个转换器:
class MyObjectConverter implements org.springframework...Converter<String[], MyObject> {
/* ... */
}
文件的验证器
class UploadFileValidator implements org.springframework.validation.Validator {
/* ... */
}
我有一张表格可以上传:
<form method="post"
action="<@spring.url '/upload'/>"
enctype="multipart/form-data">
<input id="upload" type="file" name="file"/>
<input type="submit" id="uploadButton"/>
</form
但我真正想要做的就是将它们组合在一起,这样我的控制器就可以有类似
的方法@RequestMapping(method = RequestMethod.POST)
public String doUpload(
@Valid final List<MyObject> objList,
final BindingResult result) { ...}
我知道Spring MVC框架支持转换器和验证器,但我无法理解如何让它们一起工作。
答案 0 :(得分:4)
首先,我将MultipartFile包装在表单支持对象中:
public class UploadBackingForm {
private MultipartFile multipartFile;
/* ... getter/setter */
}
然后我将其绑定到我的表单:
<form method="post" enctype="multipart/form-data">
<@spring.bind "backingform.*"/>
<tr>
<td><@spring.formInput 'backingform.multipartFile' '' 'file' /></td>
<td> <button type="submit">Upload</button> </td>
</tr>
</form>
在控制器中我指定了一个验证器:
@InitBinder
public void initBinder(final DataBinder binder) {
binder.setValidator(new UploadValidator());
}
这是验证器:
public class UploadValidator implements Validator {
private final Converter<String[], MyObject> converter
= new MyObjectConverter();
@Override
public boolean supports(final Class<?> clazz) {
return UploadBackingForm.class.equals(clazz);
}
@Override
public void validate(final Object target, final Errors errors) {
final UploadBackingForm backingForm = (UploadBackingForm) target;
final MultipartFile multipartFile = backingForm.getMultipartFile();
final List<String[]> uploadData = /* parse file */
for (final String[] uploadDataRow : uploadData){
try {
converter.convert(uploadDataRow);
} catch (IllegalArgumentException e) {
errors.rejectValue("multipartFile", "line.invalid", ...);
}
}
}
}
验证器使用转换器将行项目转换为MyObj。
doPost方法现在看起来像这样:
@RequestMapping(method = RequestMethod.POST)
public String doUpload(
@Valid @ModelAttribute("backingform") UploadBackingForm backingForm,
final BindingResult result,
final HttpSession session) throws IOException {
final UploadConverter converter = new UploadConverter();
final List<MyObj> imports =
converter.convert(backingForm.getMultipartFile().getInputStream());
}
UploadConverter与UploadValidator非常相似:
public class UploadConverter implements Converter<InputStream, List<MyObject>> {
private final Converter<String[], MyObject> converter = new MyObjectConverter();
@Override
public List<MyObject> convert(final InputStream source) {
final List<String[]> detailLines = /* ... getDetailLines */
final List<MyObject> importList =
new ArrayList<MyObject>(detailLines.size());
for (final String[] row : detailLines) {
importList.add(converter.convert(row));
}
return importList;
}
}
唯一的问题是验证和转换过程是完全相同的。幸运的是,上传文件不会很大,因此重复工作不是一个大问题。
答案 1 :(得分:0)
您需要一个能够将MultipartFile转换为MyObjects
但不能将String[]
转换为MyObjectsContainer
的转换器。 - MyObjectsContainer
只不过是MyObjects
列表的包装器。
但我真的不知道这个转换器是否可行,因为MultipartFile是一个非常特殊的参数。
对于验证我强烈建议MyObjectsContainer
然后你可以写:
@RequestMapping(method = RequestMethod.POST)
public String doUpload(@RequestParam("file") final MyObjectsContainer container) {}