Springファイルアップロードインターフェースの学習(MultipartFile、MultiparHttpservletRequest、MultipartResolver)

原文链接:https://docs.spring.io/spring/docs/current/javadoc-api/org/springframework/web/multipart/MultipartFile.html

同社のファイルアップロードはMongoDBが作成したファイルサービスをベースにしていますが、アップロードしたファイル解析データに問題があり、圧倒されたので整理して勉強しました。

抜粋在我看来,春天里一颗小草的生长,它没有什么目的。风起时一匹公马发情,它也没有什么目的...我要抱着草长马发情的伟大真诚去做一切事,而不是在人前羞羞答答的表演。 ——《三十而立》

ファイルアップロードインターフェースの理論:

  • MultiparHttpservletRequest:アップロードされたファイルへのアクセスを許可するために、サーブレット要求の複数の部分を処理する他のメソッドを提供します。実装では、マルチパートパラメータを使用可能にするために、パラメータアクセスのためにServletRequestによって使用される標準メソッドをオーバーライドする必要もあります。
 public ModelAndView handleRequest(HttpServletRequest request, HttpServletResponse response) {
    
    
   MultipartHttpServletRequest multipartRequest = (MultipartHttpServletRequest) request;
   MultipartFile multipartFile = multipartRequest.getFile("image");
   ...
 }

具体化されているDefaultMultipartHttpServletRequest中間ステップとしてAbstractMultipartHttpServletRequestサブクラスを行ってもよいです


+ DefaultMultipartHttpServletRequest:MultipartHttpServletRequest 接口的默认实现提供预生成参数值的管理CommonsMultipartResolverによって使用されます。

public class DefaultMultipartHttpServletRequest
extends AbstractMultipartHttpServletRequest
コンストラクタ 説明
DefaultMultipartHttpServletRequest(HttpServletRequest request) 指定されたHttpServletRequestをMultipartHttpServletRequestでラップします。
DefaultMultipartHttpServletRequest(HttpServletRequest request、MultiValueMap <String、MultipartFile> mpFiles、Map <String、String []> mpParams、Map <String、String> mpParamContentTypes) 指定されたHttpServletRequestをMultipartHttpServletRequestでラップします。

方法 説明
String getMultipartContentType(String paramOrFileName) 指定されたリクエストパーツのコンテンツタイプを決定します。
HttpHeaders getMultipartHeaders(String paramOrFileName) マルチパートリクエストの指定された部分に関連付けられたヘッダーを返します。
保護されたMap <String、String> getMultipartParameterContentTypes() 検索に使用するマルチパートパラメータコンテンツタイプMapを取得します。必要に応じて、遅延初期化します。
保護されたMap <String、String []> getMultipartParameters() 取得するマルチパートパラメータMapを取得し、必要に応じて、遅延して初期化します。
文字列getParameter(文字列名) このメソッドのデフォルトの動作は、ラップされたリクエストオブジェクトでgetParameter(String name)を返すことです。
Map <String、String []> getParameterMap() このメソッドのデフォルトの動作は、ラップされたリクエストオブジェクトでgetParameterMap()を返すことです。
列挙getParameterNames() このメソッドのデフォルトの動作は、ラップされたリクエストオブジェクトでgetParameterNames()を返すことです。
String [] getParameterValues(String name) このメソッドのデフォルトの動作は、ラップされたリクエストオブジェクトでgetParameterValues(文字列名)を返すことです。
protected void setMultipartParameterContentTypes(Map <String、String> multipartParameterContentTypes) パラメータ名をキー、コンテンツタイプ文字列を値としてマッピングを設定します。
protected void setMultipartParameters(Map <String、String []> multipartParameters) パラメータ名をキー、文字列配列オブジェクトを値としてマッピングを設定します。
String getMultipartContentType(String paramOrFileName) 指定されたリクエストパーツのコンテンツタイプを決定します。
HttpHeaders getMultipartHeaders(String paramOrFileName) マルチパートリクエストの指定された部分に関連付けられたヘッダーを返します。
保護されたMap <String、String> getMultipartParameterContentTypes() 検索に使用するマルチパートパラメータコンテンツタイプMapを取得します。必要に応じて、遅延初期化します。
保護されたMap <String、String []> getMultipartParameters() 取得するマルチパートパラメータMapを取得し、必要に応じて、遅延して初期化します。
文字列getParameter(文字列名) このメソッドのデフォルトの動作は、ラップされたリクエストオブジェクトでgetParameter(String name)を返すことです。
Map <String、String []> getParameterMap() このメソッドのデフォルトの動作は、ラップされたリクエストオブジェクトでgetParameterMap()を返すことです。
列挙getParameterNames() このメソッドのデフォルトの動作は、ラップされたリクエストオブジェクトでgetParameterNames()を返すことです。
String [] getParameterValues(String name) このメソッドのデフォルトの動作は、ラップされたリクエストオブジェクトでgetParameterValues(文字列名)を返すことです。
protected void setMultipartParameterContentTypes(Map<String,String> multipartParameterContentTypes) 设置一个映射,将参数名称作为键,将内容类型字符串作为值。
protected void setMultipartParameters(Map<String,String[]> multipartParameters) 设置一个映射,将参数名称作为键,将字符串数组对象作为值。

  • AbstractMultipartHttpServletRequest:MultipartHttpServletRequest接口的抽象基础实现。提供对预生成的MultipartFile实例的管理。
public abstract class AbstractMultipartHttpServletRequest
extends HttpServletRequestWrapper
implements MultipartHttpServletRequest
修饰符 构造函数和描述
protected AbstractMultipartHttpServletRequest(HttpServletRequest request) 将给定的HttpServletRequest包装在MultipartHttpServletRequest中。

修饰符和类型 方法和说明
MultipartFile getFile(String name) 返回此请求中的内容以及上载文件的描述,或者null如果不存在,则返回描述。
Map<String,MultipartFile> getFileMap() 返回Map此请求中包含的多部分文件。
Iterator getFileNames() 返回一个IteratorString对象,其中包含此请求中包含的多部分文件的参数名称。
List getFiles(String name) 返回此请求中的内容以及上载文件的描述,如果不存在,则返回一个空列表。
MultiValueMap<String,MultipartFile> getMultiFileMap() 返回MultiValueMap此请求中包含的多部分文件。
protected MultiValueMap<String,MultipartFile> getMultipartFiles() 获取要检索的MultipartFile映射,并在必要时延迟对其进行初始化。
HttpServletRequest getRequest() 返回包装的请求对象。
HttpHeaders getRequestHeaders() 以方便的HttpHeaders实例的形式返回此请求的标头。
HttpMethod getRequestMethod() 以方便的HttpMethod实例的形式返回此请求的方法。
protected void initializeMultipart() 如果可能,延迟初始化多部分请求。
boolean isResolved() 确定基础的多部分请求是否已解决。
protected void setMultipartFiles(MultiValueMap<String,MultipartFile> multipartFiles) 设置一个映射,将参数名称作为键,将MultipartFile对象的列表作为值。

  • MultipartResolver:MultipartResolver 用于处理文件上传,当收到请求时 DispatcherServlet 的 checkMultipart() 方法会调用 MultipartResolver 的 isMultipart() 方法判断请求中是否包含文件。如果请求数据中包含文件,则调用 MultipartResolver 的 resolveMultipart() 方法对请求的数据进行解析,然后将文件数据解析成 MultipartFile 并封装在 MultipartHttpServletRequest (继承了 HttpServletRequest) 对象中,最后传递给 Controller,

Spring没有默认的解析器实现 DispatcherServlets,因为应用程序可能会选择自行解析其多部分请求。要定义实现,请在DispatcherServlet’s 应用程序上下文中创建一个ID为“ multipartResolver”的bean 。这样的解析器将应用到该解析器处理的所有请求DispatcherServlet。
如果a DispatcherServlet检测到多部分请求,它将通过已配置解析该请求MultipartResolver 并传递包装的HttpServletRequest。然后,控制器可以将给定的请求转换为MultipartHttpServletRequest 接口,从而可以访问any MultipartFiles。请注意,仅在实际的多部分请求的情况下才支持此转换。几乎不需要MultipartResolver 从应用程序代码访问自身。它只会在后台进行工作,MultipartHttpServletRequests 提供给控制器。


  • MultipartFile:多部分请求中收到的上载文件的表示形式。文件内容要么存储在内存中,要么临时存储在磁盘上。在这两种情况下,用户都有责任根据需要将文件内容复制到会话级或持久性存储中。临时存储将在请求处理结束时清除。
方法 说明
byte[] getBytes() 以字节数组形式返回文件的内容。
String getContentType() 返回文件的内容类型。
InputStream getInputStream() 返回一个InputStream,以从中读取文件的内容。
String getName() 以多部分形式返回参数名称。
String getOriginalFilename() 返回客户端文件系统中的原始文件名。
default Resource getResource() 返回此MultipartFile的资源表示形式。
long getSize() 返回文件的大小(以字节为单位)。
boolean isEmpty() 返回上载的文件是否为空,即,是否没有以多部分形式选择文件或选择的文件没有内容。
void transferTo(File dest) 将收到的文件传输到给定的目标文件。
default void transferTo(Path dest) 将收到的文件传输到给定的目标文件。

Demo

package com.example.demo.fileup;
 
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.util.FileCopyUtils;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;
import org.springframework.web.multipart.MultipartFile;
 
 
import javax.servlet.http.HttpServletRequest;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.text.SimpleDateFormat;
import java.util.*;
 
/**
 * @Description :
 * @Author: Liruilong
 * @Date: 2019/10/4 12:12
 */
@Controller
public class fileupController {
    
    
 
    @Value("${file.path}")
    private String filePath;
 
    @ResponseBody
    @RequestMapping("/upfile")
    public String upfile(MultipartFile multipartFile) throws IOException {
    
    
        if (multipartFile.isEmpty()){
    
    
            System.out.println("没有传入数据");
        }
        String filenamefix = multipartFile.getOriginalFilename().substring(multipartFile.getOriginalFilename().lastIndexOf("."));
        String newFile = UUID.randomUUID().toString() + filenamefix;
        FileCopyUtils.copy(multipartFile.getInputStream(),new FileOutputStream(new File(filePath+newFile)));
        return "文件已经上传啦!!!";
    }
    // 分文件夹存取
    SimpleDateFormat sdf = new SimpleDateFormat("/yyyy/MM/dd/");
    @ResponseBody
    @PostMapping("/upload")
    public String upload(MultipartFile multipartFile, HttpServletRequest request) throws IOException {
    
    
        // 将文件存入指定的文件夹
        String format = sdf.format(new Date());
        String realPath = request.getServletContext().getRealPath("/img") + format;
        // 创建文件存放路径
        File folder = new File(realPath);
        folder.mkdir();
        if (!folder.exists()){
    
    
            folder.mkdirs();
        }
        String oldName = multipartFile.getOriginalFilename();
        String newName = UUID.randomUUID().toString() + oldName.substring(oldName.lastIndexOf("."));
        // 将文件移动到指定的文件夹.
        File file = new File(folder,newName);
        System.out.println(file.getName());
        multipartFile.transferTo(new File(folder, newName));
        return newName;
    }
 
    // 多文件上传
    @ResponseBody
    @PostMapping("/uploads")
    public String uploads(MultipartFile[] multipartFiles,HttpServletRequest request){
    
    
        // 创建文件路径
        String fommad = sdf.format(new Date());
        String filepath = request.getServletContext().getRealPath("/img") + fommad;
        File file = new File(filepath);
        if (!file.exists()){
    
    
            file.mkdirs();
        }
        Arrays.stream(multipartFiles).forEach(
                (MultipartFile multipart) ->{
    
    
                    String filename = multipart.getOriginalFilename();
                    String newName = UUID.randomUUID().toString() + filename.substring(filename.lastIndexOf("."));
                    try {
    
    
                        multipart.transferTo(new File(file, newName));
                    } catch (IOException e) {
    
    
                        e.printStackTrace();
                    }
                }
        );
        return "上传成功";
    }
}

//UUID.randomUUID().toString()是javaJDK提供的一个自动生成主键的方法。UUID(Universally Unique Identifier)全局唯一标识符,
// 是指在一台机器上生成的数字,它保证对在同一时空中的所有机器都是唯一的,是由一个十六位的数字组成,表现出来的形式。
// 由以下几部分的组合:当前日期和时间(UUID的第一个部分与时间有关,如果你在生成一个UUID之后,过几秒又生成一个UUID,则第一个部分不同,其余相同),
// 时钟序列,全局唯一的IEEE机器识别号(如果有网卡,从网卡获得,没有网卡以其他方式获得),UUID的唯一缺陷在于生成的结果串会比较长。

/**
	 * <per>
	 * <p>上传文件</p>
	 * <per/>
	 *
	 * @param request
	 * @param response
	 * @param isUseGridFS
	 * @return java.lang.Object
	 * @throws
	 * @Description : TODO 上传文件
	 * @author Liruilong
	 * @Date 2020/9/17 12:42
	 **/
public Object uploadFile(DefaultMultipartHttpServletRequest request,
			HttpServletResponse response, boolean isUseGridFS) throws Exception {
    
    
		//获取用户设置的参数
		Map<String, String[]> params = request.getParameterMap();
		/*********说明是断点续传************/
		String range = request.getHeader("RANGE");
		if(StringUtils.isNotBlank(range)){
    
    
			return uploadPointFile(request, response);
		}
		Iterator<String> uriFileNames = request.getFileMap().keySet().iterator();
		//遍历记录文件信息,并保存文件
		String urifileName = "";
		while (uriFileNames.hasNext()) {
    
    
			urifileName = uriFileNames.next();
			List<MultipartFile> mfiles = request.getFiles(urifileName);
			for(MultipartFile mFile : mfiles){
    
    			

おすすめ

転載: blog.csdn.net/sanhewuyang/article/details/108622574