麻豆小视频在线观看_中文黄色一级片_久久久成人精品_成片免费观看视频大全_午夜精品久久久久久久99热浪潮_成人一区二区三区四区

首頁 > 開發(fā) > Java > 正文

SpringBoot+fileUpload獲取文件上傳進度

2024-07-14 08:42:10
字體:
供稿:網(wǎng)友

我本人在網(wǎng)上找了很多關(guān)于文件上傳進度獲取的文章,普遍基于spring MVC 框架通過 fileUpload 實現(xiàn),對于spring Boot 通過 fileUpload 實現(xiàn)的帖子非常少,由于小弟學(xué)藝不精,雖然 Spring Boot 和 Spring MVC 相差不大,只是配置方式的差別,還是搞了很久,上傳此文章的目的是希望自己作為文本保留,以便日后查看備忘,并且希望通過我的例子可以幫助到其他人而已,如果各位大佬發(fā)現(xiàn)小弟對于某些知識有誤解,還請不吝賜教,先謝謝各位前輩了!

寫此篇文章之前我查了很多關(guān)于spring MVC 框架通過 fileUpload 實現(xiàn)進度條的帖子和文章,在此對各位作者表示感謝!

本功能基于commons fileUpload 組件實現(xiàn)

1.首先,不能在程序中直接使用 fileUpload.parseRequest(request)的方式來獲取 request 請求中的 multipartFile 文件對象,原因是因為在 spring 默認的文件上傳處理器 multipartResolver 指向的類CommonsMultipartResolver 中就是通過 commons fileUpload 組件實現(xiàn)的文件獲取,因此,在代碼中再次使用該方法,是獲取不到文件對象的,因為此時的 request 對象是不包含文件的,它已經(jīng)被CommonsMultipartResolver 類解析處理并轉(zhuǎn)型。

CommonsMultipartResolver 類中相關(guān)源碼片段:

protected MultipartParsingResult parseRequest(HttpServletRequest request) throws MultipartException {    String encoding = determineEncoding(request);    FileUpload fileUpload = prepareFileUpload(encoding);    try {      List<FileItem> fileItems = ((ServletFileUpload) fileUpload).parseRequest(request);      return parseFileItems(fileItems, encoding);    }    catch (FileUploadBase.SizeLimitExceededException ex) {      throw new MaxUploadSizeExceededException(fileUpload.getSizeMax(), ex);    }    catch (FileUploadBase.FileSizeLimitExceededException ex) {      throw new MaxUploadSizeExceededException(fileUpload.getFileSizeMax(), ex);    }    catch (FileUploadException ex) {      throw new MultipartException("Failed to parse multipart servlet request", ex);    }}

2.由于spring 中的 CommonsMultipartResolver 類中并沒有加入 processListener 文件上傳進度監(jiān)聽器,所以,直接使用 CommonsMultipartResolver 類是無法監(jiān)聽文件上傳進度的,如果我們需要獲取文件上傳進度,就需要繼承 CommonsMultipartResolver 類并重寫 parseRequest 方法,在此之前,我們需要創(chuàng)建一個實現(xiàn)了 processListener 接口的實現(xiàn)類用于監(jiān)聽文件上傳進度。

processListener接口實現(xiàn)類:

import javax.servlet.http.HttpSession;import org.apache.commons.fileupload.ProgressListener;import org.springframework.stereotype.Component;@Componentpublic class UploadProgressListener implements ProgressListener{  private HttpSession session;   public void setSession(HttpSession session){     this.session=session;     ProgressEntity status = new ProgressEntity();     session.setAttribute("status", status);   }   /*    * pBytesRead 到目前為止讀取文件的比特數(shù) pContentLength 文件總大小 pItems 目前正在讀取第幾個文件    */   @Override  public void update(long pBytesRead, long pContentLength, int pItems) {     ProgressEntity status = (ProgressEntity) session.getAttribute("status");     status.setpBytesRead(pBytesRead);     status.setpContentLength(pContentLength);     status.setpItems(pItems);   } }

ProgressEntity 實體類:

import org.springframework.stereotype.Component;@Componentpublic class ProgressEntity {   private long pBytesRead = 0L;  //到目前為止讀取文件的比特數(shù)    private long pContentLength = 0L;  //文件總大小    private int pItems;        //目前正在讀取第幾個文件   public long getpBytesRead() {     return pBytesRead;   }   public void setpBytesRead(long pBytesRead) {     this.pBytesRead = pBytesRead;   }   public long getpContentLength() {     return pContentLength;   }   public void setpContentLength(long pContentLength) {     this.pContentLength = pContentLength;   }   public int getpItems() {     return pItems;   }   public void setpItems(int pItems) {     this.pItems = pItems;   }   @Override   public String toString() {     float tmp = (float)pBytesRead;     float result = tmp/pContentLength*100;     return "ProgressEntity [pBytesRead=" + pBytesRead + ", pContentLength="         + pContentLength + ", percentage=" + result + "% , pItems=" + pItems + "]";   } } 

最后,是繼承 CommonsMultipartResolver 類的自定義文件上傳處理類:

 

import java.util.List; import javax.servlet.http.HttpServletRequest; import org.apache.commons.fileupload.FileItem; import org.apache.commons.fileupload.FileUpload; import org.apache.commons.fileupload.FileUploadBase; import org.apache.commons.fileupload.FileUploadException; import org.apache.commons.fileupload.servlet.ServletFileUpload; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.web.multipart.MaxUploadSizeExceededException; import org.springframework.web.multipart.MultipartException; import org.springframework.web.multipart.commons.CommonsMultipartResolver; public class CustomMultipartResolver extends CommonsMultipartResolver{  @Autowired  private UploadProgressListener uploadProgressListener;  @Override  protected MultipartParsingResult parseRequest(HttpServletRequest request) throws MultipartException {    String encoding = determineEncoding(request);    FileUpload fileUpload = prepareFileUpload(encoding);    uploadProgressListener.setSession(request.getSession());//問文件上傳進度監(jiān)聽器設(shè)置session用于存儲上傳進度    fileUpload.setProgressListener(uploadProgressListener);//將文件上傳進度監(jiān)聽器加入到 fileUpload 中    try {      List<FileItem> fileItems = ((ServletFileUpload) fileUpload).parseRequest(request);      return parseFileItems(fileItems, encoding);    }    catch (FileUploadBase.SizeLimitExceededException ex) {      throw new MaxUploadSizeExceededException(fileUpload.getSizeMax(), ex);    }    catch (FileUploadBase.FileSizeLimitExceededException ex) {      throw new MaxUploadSizeExceededException(fileUpload.getFileSizeMax(), ex);    }    catch (FileUploadException ex) {      throw new MultipartException("Failed to parse multipart servlet request", ex);    }  }}

3.此時,所有需要的類已經(jīng)準(zhǔn)備好,接下來我們需要將 spring 默認的文件上傳處理類取消自動配置,并將 multipartResolver 指向我們剛剛創(chuàng)建好的繼承 CommonsMultipartResolver 類的自定義文件上傳處理類。

import org.springframework.boot.SpringApplication;import org.springframework.boot.autoconfigure.EnableAutoConfiguration;import org.springframework.boot.autoconfigure.web.MultipartAutoConfiguration;import org.springframework.boot.web.servlet.ServletComponentScan;import org.springframework.context.annotation.Bean;import org.springframework.context.annotation.ComponentScan;import org.springframework.context.annotation.Configuration;import org.springframework.web.multipart.MultipartResolver;import com.example.listener.CustomMultipartResolver;/* * 將 spring 默認的文件上傳處理類取消自動配置,這一步很重要,沒有這一步,當(dāng)multipartResolver重新指向了我們定義好 * 的新的文件上傳處理類后,前臺傳回的 file 文件在后臺獲取會是空,加上這句話就好了,推測不加這句話,spring 依然 * 會先走默認的文件處理流程并修改request對象,再執(zhí)行我們定義的文件處理類。(這只是個人推測) * exclude表示自動配置時不包括Multipart配置 */@EnableAutoConfiguration(exclude = {MultipartAutoConfiguration.class})@Configuration@ComponentScan(basePackages = {"com.example"})@ServletComponentScan(basePackages = {"com.example"})public class UploadProgressApplication {/* * 將 multipartResolver 指向我們剛剛創(chuàng)建好的繼承 CommonsMultipartResolver 類的自定義文件上傳處理類 */@Bean(name = "multipartResolver")public MultipartResolver multipartResolver() {  CustomMultipartResolver customMultipartResolver = new CustomMultipartResolver();  return customMultipartResolver;}public static void main(String[] args) {  SpringApplication.run(UploadProgressApplication.class, args);}}

至此,準(zhǔn)備工作完成,我們再創(chuàng)建一個測試用的 controller 和 html 頁面用于文件上傳。 

controller:

import org.springframework.stereotype.Controller;import org.springframework.web.bind.annotation.RequestMapping;import org.springframework.web.bind.annotation.RequestMethod;import org.springframework.web.bind.annotation.ResponseBody;import org.springframework.web.multipart.MultipartFile;import org.springframework.web.servlet.ModelAndView;@Controller@RequestMapping("/uploadProgress")public class UploadController {  @RequestMapping(value = "/showUpload", method = RequestMethod.GET)  public ModelAndView showUpload() {    return new ModelAndView("/UploadProgressDemo");  }  @RequestMapping("/upload")  @ResponseBody  public void uploadFile(MultipartFile file) {    System.out.println(file.getOriginalFilename());  }}

HTML:

<!DOCTYPE html><html><head>  <meta charset="UTF-8"></meta>  <title>測試</title>這里寫代碼片</head><body>  這是文件上傳頁面  <form action="/uploadProgress/upload" method="POST" enctype="multipart/form-data">    <input type="file" name="file"/>    <br/>    <input type="submit" value="提交"/>  </form></body></html>

經(jīng)本人測試,確實可以獲取文件上傳進度,前臺頁面修改進度條進度可以采用前臺頁面輪詢的方式訪問后臺,在相應(yīng)action中通過存儲在session中的對象 status 來獲取最新的上傳進度并返回展示即可。

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持VeVb武林網(wǎng)。


注:相關(guān)教程知識閱讀請移步到JAVA教程頻道。
發(fā)表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發(fā)表
主站蜘蛛池模板: 国产一区二区精品免费 | 欧美一级黄色网 | 久草在线观看资源 | av在线播放地址 | 成人午夜淫片a | 特级毛片a级毛片100免费 | 茄子福利视频 | fc2国产成人免费视频 | 久久久久亚洲国产精品 | 99精品视频久久精品视频 | 男女无套免费视频 | 成人精品| 99国产精品白浆在线观看免费 | 在线观看国产一区二区三区 | 久久2019中文字幕 | 香蕉视频99 | 高清在线国产 | 激情网站免费观看 | 国产精品自拍99 | 精品一区久久久 | 亚洲一区动漫 | 成人免费福利视频 | 日韩激情一区 | 亚洲精品成人久久久 | 亚洲电影在线播放 | 99精品视频一区二区 | 久久精品欧美一区 | 性欧美视频在线观看 | 亚洲午夜国产 | 国产又粗又爽又深的免费视频 | 久久国产精品久久久久 | 国产日韩大片 | 国产一区免费视频 | 最新av在线播放 | 亚洲欧美日韩一区二区三区在线观看 | 亚洲欧美国产高清va在线播放 | 超碰97国产在线 | 国产91九色在线播放 | 欧美一级在线免费 | 91麻豆蜜桃一区二区三区 | 色人阁五月天 |