一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務(wù)器之家:專注于服務(wù)器技術(shù)及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術(shù)|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務(wù)器之家 - 編程語言 - Java教程 - struts2實現(xiàn)文件上傳顯示進度條效果

struts2實現(xiàn)文件上傳顯示進度條效果

2020-09-24 15:45筱筱清流水 Java教程

這篇文章主要為大家詳細介紹了struts2實現(xiàn)文件上傳顯示進度條效果,具有一定的參考價值,感興趣的小伙伴們可以參考一下

一. struts2讀取進度原理分析(作為草稿存了好久,剛剛發(fā)布出來......)

1. 在strut2中控制文件上傳信息的類是實現(xiàn)MultiPartRequest接口的JakartaMultiPartRequest

其實第一次看到源文件時我打了個退堂鼓,因為覺得內(nèi)容太長了,不想看。冷靜下來將思路理順,將分開的各個方法還原到一個方方中中,發(fā)現(xiàn)還是很好理解的:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
@Override
 public void parse(HttpServletRequest request, String saveDir)
   throws IOException {
  setLocale(request);
     //規(guī)定了File文件的格式(如文件名必須是xxFileName,文件類型xxContentType),并定義了File的保存路徑                 DiskFileItemFactory factory = new DiskFileItemFactory();
  ServletFileUpload upload = new ServletFileUpload(factory);//處理文件上傳的servlet
  upload.setProgressListener(new FileUploadProgressListener(request)); //為文件上傳添加監(jiān)聽  factory.setSizeThreshold(0); //if (saveDir != null
   factory.setRepository(new File(saveDir));//臨時路徑
  }
  try {
   upload.setSizeMax(maxSize);
   List items = upload.parseRequest(createRequestContext(request)); //獲取所有請求
   for (Object obItem : items) {
    FileItem item = (FileItem) obItem; //獲取每個請求的文件
    if (LOG.isDebugEnabled()) {
     LOG.debug("Found item" + item.getFieldName());
    }
    if (item.isFormField()) { //普通表單提交
     LOG.debug("Item is a normal form field");
     List<String> values;
     if (params.get(item.getFieldName()) != null) {
      values = params.get(item.getFieldName());
     } else {
      values = new ArrayList<String>();
     }
     String charset = request.getCharacterEncoding();
     if (charset != null) {
      values.add(item.getString(charset));
     } else {
      values.add(item.getString());
     }
     params.put(item.getFieldName(), values);
    } else { //文件上傳請求
     LOG.debug("Item is a file upload");
     if (item.getName() == null
       || item.getName().trim().length() <= 0) {
      LOG.debug("No file has been uploded for the filed:"
        + item.getFieldName());
      continue;
     }
 
     List<FileItem> values;
     if (files.get(item.getFieldName()) != null) {
      values = files.get(item.getFieldName());
     } else {
      values = new ArrayList<FileItem>();
     }
     values.add(item);
     files.put(item.getFieldName(), values);
    }
   }
 
  } catch (FileUploadBase.SizeLimitExceededException e) {
   System.out.println("錯誤1:" + e);
   if (LOG.isWarnEnabled()) {
    LOG.warn("Request exceeded size limit!", e);
   }
   String errorMessage = buildErrorMessage(e, new Object[]{e.getPermittedSize(), e.getActualSize()});
   if (!errors.contains(errorMessage)) {
    errors.add(errorMessage);
   }
  } catch (Exception e) {
   System.out.println("錯誤1:" + e);
   if (LOG.isWarnEnabled()) {
    LOG.warn("Unable to parse request", e);
   }
   String errorMessage = buildErrorMessage(e, new Object[]{});
   if (!errors.contains(errorMessage)) {
    errors.add(errorMessage);
   }
  }
 }

2.  文件上傳監(jiān)聽文件FileUploadProgressListener.java

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
public class FileUploadProgressListener implements ProgressListener {
  private final HttpSession session;
  private final DecimalFormat format = new DecimalFormat("#00.0");
 
  public FileUploadProgressListener(HttpServletRequest request) {
    session = request.getSession();
    FileUploadStatus status = new FileUploadStatus();
    session.setAttribute("uploadStatus", status);
  }
 
  @Override
  public void update(long pBytesRead, long pContentLength, int pItems) {
    FileUploadStatus uploadStatus = (FileUploadStatus) session.getAttribute("uploadStatus");
    Double uploadRate = (double) (pBytesRead * 100 / pContentLength);
    uploadStatus.setUploadRate(Double.valueOf(format.format(uploadRate)));
    uploadStatus.setReadedBytes(pBytesRead / 1024);
    uploadStatus.setTotalBytes(pContentLength / 1024);
    uploadStatus.setCurrentItems(pItems);
  }
}

3. 添加狀態(tài)文件:FileUploadStatus.java

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
public class FileUploadStatus {
 private Double uploadRate = 0.0;
 private Long readedBytes = 0L;
 private Long totalBytes = 0L;
 private int currentItems = 0;
 private Long uploadSpeed = 0L;
 private Long startTime = System.currentTimeMillis();
 private Long readedTimes = 0L;
 private Long totalTimes = 0L;
 // "-1" 錯誤 "0" 正常 "1" 完成
 private String error = "0";
 
 ...
  setter getter方法
 ... 
}

4. Action類(如果是多文件上傳,則將File   FileName   ContentType定義成數(shù)組形式即可)

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
/**
 * 利用io流上傳文件
 */
public class FileStreamUploadAction extends ActionSupport {
 /**
  * serialVersionUID作用: ---相當于類的身份證。 序列化時為了保持版本的兼容性,即在版本升級時反序列化仍保持對象的唯一性。
  * 有兩種生成方式: 一個是默認的1L,比如:private static final long serialVersionUID = 1L;
  * 一個是根據(jù)類名、接口名、成員方法及屬性等來生成一個64位的哈希字段,比如: private static final long
  * serialVersionUID = xxxxL;
  */
 private static final long serialVersionUID = 1L;
 private File image;
 private String imageFileName;
 private String imageContentType;
 private String message;
 public String uploadFile() {
  FileInputStream in = null;
  FileOutputStream out = null;
  System.out.println("文件名:" + imageFileName);
  try {
   this.setNewFileName(imageFileName);
   String realPath = ServletActionContext.getServletContext()
     .getRealPath("/file");
   File filePath = new File(realPath);
   if (!filePath.exists()) { // 如果保存的路徑不存在則創(chuàng)建
    filePath.mkdir();
   }
   if (image == null) {
    message = "上傳文件為空";
    System.out.println(message);
   } else {
    File saveFile = new File(filePath, this.getNewFileName());
    out = new FileOutputStream(saveFile);
   }
   in = new FileInputStream(image);
   byte[] byt = new byte[1024];
   int length = 0;
   while ((length = in.read(byt)) > 0) {
    out.write(byt, 0, length);
    out.flush();
   }
   message = "上傳成功";
   System.out.println(message);
  } catch (FileNotFoundException e) {
   message = "找不到文件!";
   e.printStackTrace();
  } catch (IOException e) {
   message = "文件讀取失敗!";
   e.printStackTrace();
  } finally {
   closeStream(in, out);
  }
  return "uploadSucc";
 }
 public void closeStream(FileInputStream in, FileOutputStream out) {
  try {
   if (in != null) {
    in.close();
   }
   if (out != null) {
    out.close();
   }
  } catch (IOException e) {
   // TODO Auto-generated catch block
   e.printStackTrace();
  }
 }
  ...
  setter() getter()
  ... 
}

獲取進度的Action

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
public class FileProgressAction extends ActionSupport {
  private static final long serialVersionUID = 1L;
  private FileUploadStatus uploadStatus;
 
  public String uploadPercent() {
    HttpSession session = ServletActionContext.getRequest().getSession();
    this.uploadStatus = (FileUploadStatus) session.getAttribute("uploadStatus");
    if (uploadStatus == null) {
      System.out.println("action is null");
      uploadStatus = new FileUploadStatus();
      uploadStatus.setCurrentItems(0);
    }
    return "getPercent";
  }
 
  public FileUploadStatus getUploadStatus() {
    return uploadStatus;
  }
 
  public void setUploadStatus(FileUploadStatus uploadStatus) {
    this.uploadStatus = uploadStatus;
  }
}

5.struts.xml中

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
<struts>
  <constant name="struts.multipart.maxSize" value="2147483648"/><!-- 默認值為2M,設(shè)置為2G -->
  <constant name="struts.custom.i18n.resources" value="messageResource" />
  <constant name="struts.i18n.encoding" value="utf-8" />
  <constant name="struts.multipart.saveDir" value="e:/fileUpload"/><!-- 臨時路徑 -->
 
  <!-- 加載自定義的文件讀取配置文件 -->
  <bean type="org.apache.struts2.dispatcher.multipart.MultiPartRequest" name="Refactor" class="com.nova.core.RefactorMultiPartRequest" scope="default" />
  <constant name="struts.multipart.handler" value="Refactor" />
  <!-- 這里配置struts.multipart.handler -->
  <package name="ajaxUpload" extends="json-default"> <!-- json-default需要struts2-json-plugin-2.3.3.jar -->
   <action name="ajaxUploadFile_*" class="com.nova.action.FileStreamUploadAction" method="{1}">
    <result type="json" name="uploadSucc">
     <param name="root">newFileName</param>
     <param name="contentType">
      text/html
     </param>
    </result>
   </action>
   <action name="uploadPercent_*" class="com.nova.action.FileProgressAction" method="{1}">
    <result name="getPercent" type="json">
     <param name="root">uploadStatus</param>
    </result>
   </action>
  </package>
 </struts>

二.  進度條顯示

View頁面設(shè)置,利用ajaxfileupload.js來獲取文件并進行異步上傳,bootstrap中的進度條效果顯示進度(利用setInterval間斷的獲取進度信息來形式一種進度的前進顯示)

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
<link rel="stylesheet" type="text/css" href="bootstrap/css/bootstrap.css" rel="external nofollow" >
<link rel="stylesheet" type="text/css" href="bootstrap/css/bootstrap-responsive.css" rel="external nofollow" >
<script type="text/javascript" src="js/jquery.js"></script>
<script type="text/javascript" src="js/ajaxfileupload.js"></script>
<script type="text/javascript" src="<%=request.getContextPath() %>/bootstrap/js/bootstrap.js"></script>
<script type="text/javascript" src="<%=request.getContextPath() %>/bootstrap/js/jquery.showLoading.min.js"></script>
<script type="text/javascript">
 var setinterval;
 $(document).ready(function(){
  $("#upload").click(function(){
   $("#upload").addClass("disabled");
   $("#upload").attr("disabled" ,true);
   $("#upload").attr("title" ,"文件上傳中...");
   uploadFile();
   setinterval = setInterval(uploadProgress,200);
  });
 });
 //文件上傳
 function uploadFile(){
  $.ajaxFileUpload({
   url:'ajaxUploadFile_uploadFile.action',
   secureuri:false, //是否采用安全協(xié)議,默認為false
   fileElementId:'image',
   dataType: 'json',
   success: function (data){
    $("#showImage").attr("src","/FileUpLoadTest/file/"+data);
   }
  });
 }
 //上傳進度
 function uploadProgress(){
  $.get("uploadPercent_uploadPercent.action","",function(data){
   $("#ProgressRate").html("上傳速度:" + data.uploadRate + "%");
   $("#readBytes").html("以讀取:" + data.readedBytes + " KB");
   $("#totalBytes").html("總大小:" + data.totalBytes + " KB");
   $("#progress").attr("style","width:" + data.uploadRate + "%;");
   $("#progress").html(data.uploadRate + "%");
   if(data.uploadRate == 100){
    clearInterval(setinterval);
    $("#progress").html("上傳成功");
    $("#upload").removeClass("disabled");
    $("#upload").attr("disabled" ,false);
   }
  });
 }
</script>
</head>
<body>
 <div class="navbar navbar-inverse navbar-fixed-top">
  <div class="navbar-inner">
  <div class="container">
   <button type="button" class="btn btn-navbar" data-toggle="collapse" data-target=".nav-collapse">
   <span class="icon-bar"></span>
   <span class="icon-bar"></span>
   <span class="icon-bar"></span>
   </button>
   <a class="brand" href="#" rel="external nofollow" >文件異步上傳+進度條</a>
  </div>
  </div>
 </div>
 <br><br><br>
 <div class="container">
  <input type="file" name="image" id="image"/><br/> //file的name屬性必須設(shè)置的與后臺Action中file的名稱是相同的,否則ajaxFileUpload獲取不到文件信息
  <input type="button" id="upload" value="上傳" class="btn btn-info" title=""/><br/>
  <img alt="" src="" id="showImage">
  <div id="ProgressRate"></div>
  <div id="readBytes"></div>
  <div id="totalBytes"></div>
  <div id="uploadTimes"></div>
  <div class="progress progress-striped span4">
    <div id="progress" class="bar">
    </div>
  </div>
 </div>
</body>
</html>

三、總結(jié)

  用這種方法獲取上傳進度有一個缺點:讀取進度階段是文件從指定目錄開始在臨時文件中存儲的過程,而文件上傳則是重臨時路徑下將文件轉(zhuǎn)移到目標路徑下,這樣就造成了一個時間差,就是讀取進度總會比上傳文件快,上傳的文件越大這個缺點越是明顯。

以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持服務(wù)器之家。

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 精品日韩欧美一区二区三区在线播放 | 四虎精品永久在线网址 | 亚洲天堂色图 | 日本在线观看www免费 | 性关系免费视频 | 欧美人与禽杂交大片 | 欧美╳bbbb | 美女被视频 | 国产xxx在线 | a片毛片在线免费看 | 帅老头恋帅老头同性tv | 动漫美女胸被狂揉扒开吃奶动态图 | 女同69式互添在线观看免费 | 手机看片一区二区 | 99亚洲| 日本免费精品视频 | 日本国产最新一区二区三区 | 91po国产在线高清福利 | 日韩在线二区全免费 | 国产精品久久久久aaaa | ass性强迫rape | 精品一区二区三区在线视频观看 | 操姓| 欧美成人禁片在线观看俄罗斯 | 免费欧美日韩 | 欧美一级免费看 | 91国产高清 | 奇米777狠狠 | 成人精品一区二区三区中文字幕 | a性片| 爽爽窝窝午夜精品一区二区 | 污污的动态图合集 | 精品亚洲欧美中文字幕在线看 | 国产夜趣福利第一视频 | 国产日韩欧美在线一区二区三区 | 久久伊人久久 | 羞羞漫画免费漫画页面在线看漫画秋蝉 | julianann办公室 | 3黑人巨大vs北岛玲 3d肉浦团在线观看 3d动漫免费 | 久久re热在线视频精99 | 糖心vlog视频永久破解版 |