struts有默认的文件拦截器,一般配置maximumSize就可以了。
知道原理,我们可以写一个类继承它,实现自己的配置上传文件大小的方式。
然后细究页面上传文件的时候,发现了一些问题。
action配置中需要三个参数:File uploadFile,String uploadFileFileName, String uploadFileContentType
然而,页面上确实没有后面两个参数的配置,只有一个文件的控件。
最后,完成上传之后,保存的时候那两个属性里却是有值的,不得而知,只有看源码,找答案。
FileUploadInterceptor
这个拦截器就是需要配置的文件上传拦截器,里面的拦截方法里面,有intercept()方法,负责对文件上传的拦截。
研究其中的代码可以看到,上传文件主要是下面这个请求在起作用:
MultiPartRequestWrapper multiWrapper = (MultiPartRequestWrapper)request;
Enumeration fileParameterNames = multiWrapper.getFileParameterNames();
File[] files = multiWrapper.getFiles(inputName);
if (files != null && files.length > 0) {
List<File> acceptedFiles = new ArrayList(files.length);
List<String> acceptedContentTypes = new ArrayList(files.length);
List<String> acceptedFileNames = new ArrayList(files.length);
String contentTypeName = inputName + "ContentType";
String fileNameName = inputName + "FileName";
for(int index = 0; index < files.length; ++index) {
if (this.acceptFile(action, files[index], fileName[index], contentType[index], inputName, validation)) {
acceptedFiles.add(files[index]);
acceptedContentTypes.add(contentType[index]);
acceptedFileNames.add(fileName[index]);
}
}
if (!acceptedFiles.isEmpty()) {
Map<String, Object> params = ac.getParameters();
params.put(inputName, acceptedFiles.toArray(new File[acceptedFiles.size()]));
params.put(contentTypeName, acceptedContentTypes.toArray(new String[acceptedContentTypes.size()]));
params.put(fileNameName, acceptedFileNames.toArray(new String[acceptedFileNames.size()]));
}
}
上面黄底的就是为什么action中的uploadFile、uploadFileFileName、uploadFileContentType会被赋值了。因为,请求的时候默认把咱们form表单里的File的name属性获取,然后把对应的文件名、内容信息等存入与文件名相关的变量中,都是默认加上“contentType”“fileName”,所以上面的问题就迎刃而解了。
此时,为什么actionContext中的getParameters中的信息,会被存放入action的对应setter、getter项中?
经过一段时间的寻根溯源,又看到一个拦截器,ParametersInterceptor(还有一个excludeParams参数用以过滤不需要赋值的参数,和order参数用以是否按顺序初始化action中的值)。
这个类有如下描述:
* This interceptor sets all parameters on the value stack.
*typically resulting in the values submitted in a form
* request being applied to an action in the value stack.
从描述中可以看到,就是通过这个拦截器,表单提交的信息可以获取,同样actionContext的getparameters方法里的所有参数也会被赋值入。
这个时候,如果细究的话,为什么valueStack中的值会被setter/getter方法知晓并赋值呢?
--最终进行参数赋值是调用的 ValueStack的setValue方法,该方法内部使用是OGNL表达式引擎进行赋值的,虽然内部非常复杂,但我们只需要知道OGNL表达式引擎在 把请求参数设置到ValueStack中时,是从栈顶往栈底寻找有相应setter方法的对象,如果正在赋值的参数在ValueStack找到了一个对象 有setter方法则把该参数的值赋给该对象,如果没有找到则继续往栈底寻找,直到找到为止,如果找到栈底还是没有找到也就没有赋值成功。
以下是valueStack中的描述,当遇到EL表达式时,会找对应的getter/setter方法,获取对应的值。
具体的更深层次的实现,还不清楚,继续看咯。(大神可以指导)
* ValueStack allows multiple beans to be pushed in and dynamic EL expressions to be evaluated against it. When evaluating an expression, the stack will be searched down the stack, from the latest objects pushed in to the earliest, looking for a bean with a getter or setter for the given property or a method of the given name (depending on the expression being evaluated).