Spring MultipartFile验证和转换

时间:2011-08-23 12:45:41

标签: spring spring-mvc

我目前有一个采用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框架支持转换器和验证器,但我无法理解如何让它们一起工作。

2 个答案:

答案 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

上的JSR 303 Bean验证注释

然后你可以写:

@RequestMapping(method = RequestMethod.POST)
public String doUpload(@RequestParam("file") final MyObjectsContainer container) {}