我刚接触到春天,我目前正在努力解决在视图中显示错误信息的多部分表单提交/验证方案所需的许多部分。
以下是我目前拥有的文件:
resourceupload.jsp:一个显示上传文件的表单的视图。
<form:form method="post" action="resource/upload" enctype="mutlipart/form-data">
<input name="name" type="text"/>
<input name="file" type="file" />
<input type="submit"/>
<form:errors path="file" cssClass="errors"/>
</form>
resourceuploadcontroller.java:处理表单提交的控制器,并且(不成功)尝试将文件验证错误发送回视图:
@RequestMapping(method = RequestMethod.POST)
public String handleFormUpload( @RequestParam("file") MultipartFile file , @RequestParam("name") String name,Object command, Errors validationErrors){
..perform some stuff with the file content, checking things in the database, etc...
.. calling validationErrors.reject("file","the error") everytime something goes wrong...
return "redirect:upload"; // redirect to the form, that should display the error messages
现在,这种方法显然有问题:
1 /我必须在validationErrors参数之前添加一个虚拟“命令”对象,否则spring会给我一个错误。这似乎不太合适。
2 /我添加了该参数后,重定向不会将错误传递给视图。我尝试在控制器的开头使用@SessionAttribute(“file”),没有任何运气。
如果有人可以提供帮助......我已经看了@ResponseBody注释,但似乎没有用于视图..
答案 0 :(得分:3)
好像我自己找到了解决方案。
首先,给我很多帮助的链接:http://www.ioncannon.net/programming/975/spring-3-file-upload-example/ 和 Spring 3 MVC - form:errors not showing the errors 这显示了使用
显示所有错误的好方法<form:errors path="*"/>.
现在,列出了我为改变这一点而改变的所有事情:
1 /使用“rejectValue”而不是“拒绝”。
2 /直接返回视图而不是重定向。
3 /创建一个带有CommonsMultipartFile属性的“UploadItem”模型
所以,总而言之,我的控制器方法变成了
@RequestMapping(method = RequestMethod.POST)
public String handleFormUpload( @ModelAttribute("uploadItem") UploadItem uploadItem, BindingResult errors){
... use errors.rejectValue ... in case of errors (moving everything i could in a UploadItemValidator.validate function)
return "uploadform"
希望有所帮助。
答案 1 :(得分:2)
这是一种非常简单的方法:
formBackingObject:
import org.springframework.web.multipart.commons.CommonsMultipartFile;
public class FileImport {
CommonsMultipartFile importFile;
public CommonsMultipartFile getImportFile() {
return importFile;
}
public void setImportFile(CommonsMultipartFile importFile) {
this.importFile = importFile;
}
}
表格:
<sf:form method="POST" modelAttribute="fileImport" enctype="multipart/form-data" action="import">
<table>
<spring:hasBindErrors name="fileImport">
<tr>
<td colspan="2">
<sf:errors path="importFile" cssClass="error"/>
</td>
</tr>
</spring:hasBindErrors>
<tr>
<th><label for="importFile">Import Workload:</label></th>
<td><input name="importFile" type="file"></td>
</tr>
</table>
<input type="submit" value="Import Application Data">
</sf:form>
最后是ControllerClassMethod:
@RequestMapping(value={"/import"}, method=RequestMethod.POST)
public String importWorkload(
FileImport fileImport, BindingResult bindResult,
@RequestParam(value="importFile", required=true) MultipartFile importFile ){
if( 0 == importFile.getSize() ){
bindResult.rejectValue("importFile","fileImport.importFile");
}
if(bindResult.hasErrors()){
return "admin";
}
....
}
轻松自在!
@NotNull注释不适用于文件的原因是因为多部分文件在请求对象中永远不为空。但是,它可以有0个字节。您可以花时间实现自定义验证器,但为什么?
答案 2 :(得分:1)
我不认为这种方法完全包含spring,但它很有效,很简单,并且使用简单的html表单,以及控制器中的一种方法:
html文件来做帖子(保存为jsp或html,你的电话)
<html>
<head>
<title>File Upload Example</title>
</head>
<body>
<form action="save_uploaded_file.html" method="post" enctype="multipart/form-data">
Choose a file to upload to the server:
<input name="myFile" type="file"/><br/>
<p>
<input type="submit"/>
<input type="reset"/>
</p>
</form>
</body>
</html>
在控制器中,请使用以下方法:
@RequestMapping("/save_uploaded_file")
public String testUpload(HttpServletRequest request) throws Exception
{
// get the file from the request
MultipartHttpServletRequest multipartRequest = (MultipartHttpServletRequest) request;
MultipartFile multipartFile = multipartRequest.getFile("myFile");
// Note: Make sure this output folder already exists!
// The following saves the file to a folder on your hard drive.
String outputFilename = "/tmp/uploaded_files/file2.txt";
OutputStream out = new FileOutputStream(outputFilename);
IOUtils.copy(multipartFile.getInputStream(), out);
return "/save_uploaded_file";
}
要使更大的文件可上载(超过40k),您可能需要在spring servlet配置中指定以下内容....我已经在那里放了一些大的#以确保它有效,但当然把如果你需要,那就是“真正的”数字。
<!-- Configure the multipart resolver -->
<bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
<!-- one of the properties available; the maximum file size in bytes -->
<property name="maxUploadSize" value="500000000"/>
<property name="maxInMemorySize" value="500000000" />
</bean>
请注意,Spring的文件上传内容使用了Apache Commons FileUpload。您需要下载并将其包含在您的应用中。反过来,这也需要Apache Commons IO,所以你也需要它。 (引自here)
答案 3 :(得分:1)
在Spring 4中,您可以实现如下文件验证器:
@Component
public class FileValidator implements Validator {
@Override
public boolean supports(Class<?> clazz) {
return FileModel.class.isAssignableFrom(clazz);
}
@Override
public void validate(Object target, Errors errors) {
FileModel fileModel = (FileModel) target;
if (fileModel.getFile() != null && fileModel.getFile().isEmpty()){
errors.rejectValue("file", "file.empty");
}
}
}
然后将上面的验证器注入上传控制器,如下所示:
@Controller
@RequestMapping("/")
public class FileUploadController {
@Autowired
private FileValidator fileValidator;
@ModelAttribute
public FileModel fileModel(){
return new FileModel();
}
@InitBinder
protected void initBinderFileModel(WebDataBinder binder) {
binder.setValidator(fileValidator);
}
@RequestMapping(method = RequestMethod.GET)
public String single(){
return "index";
}
@RequestMapping(method = RequestMethod.POST)
public String handleFormUpload(@Valid FileModel fileModel,
BindingResult result,
RedirectAttributes redirectMap) throws IOException {
if (result.hasErrors()){
return "index";
}
MultipartFile file = fileModel.getFile();
InputStream in = file.getInputStream();
File destination = new File("/tmp/" + file.getOriginalFilename());
FileUtils.copyInputStreamToFile(in, destination);
redirectMap.addFlashAttribute("filename", file.getOriginalFilename());
return "redirect:success";
}
}
此实现可帮助您的代码更清晰,更易读。我是在教程Spring MVC File Upload Validation Example
中找到的