詳解SpringBoot如何實(shí)現(xiàn)一個(gè)實(shí)時(shí)更新的進(jìn)度條

這篇文章主要詳解SpringBoot如何實(shí)現(xiàn)一個(gè)實(shí)時(shí)更新的進(jìn)度條,內(nèi)容清晰明了,對此有興趣的小伙伴可以學(xué)習(xí)一下,相信大家閱讀完之后會有幫助。

創(chuàng)新互聯(lián)自2013年創(chuàng)立以來,先為桃江等服務(wù)建站,桃江等地企業(yè),進(jìn)行企業(yè)商務(wù)咨詢服務(wù)。為桃江企業(yè)網(wǎng)站制作PC+手機(jī)+微官網(wǎng)三網(wǎng)同步一站式服務(wù)解決您的所有建站問題。

前言

博主近期接到一個(gè)任務(wù),大概內(nèi)容是:導(dǎo)入excel表格批量修改狀態(tài),期間如果發(fā)生錯(cuò)誤則所有數(shù)據(jù)不成功,為了防止重復(fù)提交,做一個(gè)類似進(jìn)度條的東東。

那么下面我會結(jié)合實(shí)際業(yè)務(wù)對這個(gè)功能進(jìn)行分析和記錄。

正文

思路

前端使用bootstrap,后端使用SpringBoot分布式到注冊中心,原先的想法是導(dǎo)入表格后異步調(diào)用修改數(shù)據(jù)狀態(tài)的方法,然后每次計(jì)算修改的進(jìn)度然后存放在session中,前臺jquery寫定時(shí)任務(wù)訪問獲取session中的進(jìn)度,更新進(jìn)度條進(jìn)度和百分比。但是這存在session在服務(wù)間不共享,跨域問題。那么換一個(gè)方式存放,存放在redis中,前臺定時(shí)任務(wù)直接操作獲取redis的數(shù)據(jù)。

實(shí)施

進(jìn)度條

先來看一下bootstrap的進(jìn)度條

<div class="progress progress-striped active">
  <div class="progress-bar progress-bar-success" role="progressbar"
     aria-valuenow="60" aria-valuemin="0" aria-valuemax="100"
     >
    40%
  </div>
</div>

 進(jìn)度條更新主要更新的值即可,div里面的40%可以省略,無非時(shí)看著明確。

可以考慮將進(jìn)度條放入彈出層。

定時(shí)任務(wù)

//點(diǎn)擊確認(rèn)導(dǎo)入執(zhí)行此方法
function bulkImportChanges() {
  //獲取批量操作狀態(tài)文件
  var files = $("#importChanges").prop("files");
  var changesFile = files[0];
  var formData = new FormData();
  formData.append("importFile",changesFile);
  $.ajax({
    type : 'post',
    url : "/risk/bulk***es",
    data : formData,
    processData : false,   //文件ajax上傳要加這兩個(gè)的,要不然上傳不了
    contentType : false,   //
    success : function(obj) {
      //導(dǎo)入成功
      if (obj.rspCode == "00") {
        //定時(shí)任務(wù)獲取redis導(dǎo)入修改進(jìn)度
        var progress = "";
        var timingTask = setInterval(function(){
          $.ajax({
            type: 'post',
            url: "/risk/t***k",
            dataType : 'json',
            success: function(result) {
              progress = result.value;
              if (progress != "error"){
                var date = progress.substring(0,6);
                //這里更新進(jìn)度條的進(jìn)度和數(shù)據(jù)
                $(".progress-bar").width(parseFloat(date)+"%");
                $(".progress-bar").text(parseFloat(date)+"%");
              }
            }
          });
          //導(dǎo)入修改完成或異常(停止定時(shí)任務(wù))
          if (parseInt(progress)==100 || progress == "error") {
            //清除定時(shí)執(zhí)行
            clearInterval(timingTask);
            $.ajax({
              type: 'post',
              url: "/risk/de***ess",
              dataType : 'json',
              success: function(result) {
                $("#bulkImportChangesProcessor").hide();
                if (parseInt(progress) == 100) {
                  alert("批量導(dǎo)入修改狀態(tài)成功");
                }
                if (progress == "error") {
                  alert("批量導(dǎo)入修改狀態(tài)失敗");
                }
                //獲取最新數(shù)據(jù)
                window.location.href="/risk/re***ByParam" rel="external nofollow" rel="external nofollow" ;
              }
            });
          }
        }, 1000)
      }else {
        $("#bulkImportChangesProcessor").hide();
        alert(obj.rspMsg);
        window.location.href="/risk/re***ByParam" rel="external nofollow" rel="external nofollow" ;
      }
    }
  });
}

解釋:點(diǎn)擊確認(rèn)導(dǎo)入文件后成功后開啟定時(shí)任務(wù)每一秒(一千毫秒)訪問一次后臺獲取redis存放的進(jìn)度,返回更新進(jìn)度條,如果更新完成或者更新失敗(根據(jù)后臺返回的數(shù)據(jù)決定)則停止定時(shí)任務(wù)顯示相應(yīng)的信息并刷新頁面。獲取最新數(shù)據(jù)。

后臺控制層

/**
 * 退單管理批量修改狀態(tài)導(dǎo)入文件
 * @param importFile
 * @return
 */
 @ResponseBody
 @RequestMapping("/bulk***es")
 public Map<String,Object> bulk***es(MultipartFile importFile){
   log.info("退單管理批量修改狀態(tài)導(dǎo)入文件,傳入?yún)?shù):"+importFile);
 Map<String,Object> map = new HashMap<>();
 List<Bulk***esEntity> fromExcel = null;
 try{
      //使用工具類導(dǎo)入轉(zhuǎn)成list
  String[] header = {"sy***um","t***mt","ha***ult","re***nd","sy***nd","r**k"};
  fromExcel = importExcelUtil.importDataFromExcel(importFile, header, BulkImportChangesEntity.class);
  if (fromExcel.size()==0){
  map.put("rspCode","99");
  map.put("rspMsg","導(dǎo)入數(shù)據(jù)不能為空");
  return map;
  }
 }catch (Exception e){
  map.put("rspCode","99");
  map.put("rspMsg","導(dǎo)入操作表失敗,請注意數(shù)據(jù)列格式");
  return map;
 }
 try {
  //這里會對list集合中的數(shù)據(jù)進(jìn)行處理

  log.info("調(diào)用服務(wù)開始,參數(shù):"+JSON.toJSONString(fromExcel));
  //String url = p4_zuul_url+"/***/ri***eat/bu***nges";
  String url = p4_zuul_url+"/***-surpass/ri***eat/bu***nges";
  String result = HttpClientUtil.doPost(url,JSON.toJSONString(fromExcel));
  log.info("調(diào)用服務(wù)結(jié)束,返回?cái)?shù)據(jù):"+result);
  if (result != null){
  map = JSONObject.parseObject(result, Map.class);
  log.info("批量修改狀態(tài)導(dǎo)入:"+JSON.toJSONString(map));
  }
 }catch (Exception e){
  map.put("rspCode","99");
  map.put("rspMsg","導(dǎo)入操作表失敗");
  log.info("bu***es exception",e);
  return map;
 }
 return map;
 }

 /**
 * 獲取退單管理批量修改狀態(tài)導(dǎo)入文件進(jìn)度條進(jìn)度
 * @return
 */
 @ResponseBody
 @RequestMapping("/t***sk")
 public Map<String,Object> t***sk(){
 Map<String,Object> map = new HashMap<>();
    //獲取redis值
 String progress = HttpClientUtil.doGet(
  p4_zuul_url + "/" + p4_redis + "/redis***ler/get&#63;key=progressSchedule");
 if (progress != null){
  map = JSONObject.parseObject(progress, Map.class);
  log.info("進(jìn)度條進(jìn)度:"+JSON.toJSONString(map));
  map.put("progressSchedule",progress);
 }else {
  HttpClientUtil.doGet(
   p4_zuul_url + "/" + p4_redis + "/redis***ler/del&#63;key=progressSchedule");
 }
 return map;
 }

 /**
 * 清除redis進(jìn)度條進(jìn)度
 * @return
 */
 @ResponseBody
 @RequestMapping("/de***ess")
 public Map<String,Object> de***ess(){
 Map<String,Object> map = new HashMap<>();
 String progress = HttpClientUtil.doGet(
  p4_zuul_url + "/" + p4_redis + "/redis***ler/del&#63;key=progressSchedule");
 if (progress != null){
  map = JSONObject.parseObject(progress, Map.class);
  log.info("返回?cái)?shù)據(jù):"+JSON.toJSONString(map));
 }
 return map;
 }

導(dǎo)入時(shí)調(diào)用第一個(gè)bulk***es方法,定時(shí)任務(wù)調(diào)用t***sk方法,導(dǎo)入完成或發(fā)生錯(cuò)誤調(diào)用de***ess方法刪除redis數(shù)據(jù),避免占用資源。

服務(wù)層

@Async//開啟異步
 @Transactional(rollbackFor = Exception.class)//事務(wù)回滾級別
 @Override
 public void bulkImportChanges(List<BulkImportChangesParam> list) {
 //初始化進(jìn)度
 Double progressBarSchedule = 0.0;
 redisClient.set("progressSchedule", progressBarSchedule + "");//存redis
 try {
  for (int i = 1; i <= list.size(); i++) {
  RiskRetreatEntity entity = riskRetreatMapper.selectRetreatListBySysRefNum(list.get(i-1).getSysRefNum());
  if (entity == null){
   //查詢結(jié)果為空直接進(jìn)行下次循環(huán)不拋出
   continue;
  }
  //實(shí)體封裝
        ···
        //更新
  riskRetreatMapper.updateRetreatByImport(entity);
  //計(jì)算修改進(jìn)度并存放redis保存(1.0 / list.size())為一條數(shù)據(jù)進(jìn)度
  progressBarSchedule = (1.0 / list.size()) * i*100;
  redisClient.set("progressSchedule", progressBarSchedule+"");
  if (i==list.size()){
   redisClient.set("progressSchedule", "100");
  }
  }
 }catch (Exception e){
  //當(dāng)發(fā)生錯(cuò)誤則清除緩存直接拋出回滾
  redisClient.set("progressSchedule","error");
  log.info("導(dǎo)入更新錯(cuò)誤,回滾");
  log.info("bulkImportChanges exception:",e);
  throw e;
 }
 }

每更新一條數(shù)據(jù)存放進(jìn)度,當(dāng)發(fā)生錯(cuò)誤則進(jìn)行回滾。如果開啟異步則需要在啟動(dòng)類添加注解@EnableAsync。

@EnableAsync
···//其他注解
public class Application {
  public static void main(String[] args) {
    SpringApplication.run(Application.class, args);
  }
}

結(jié)果樣式

詳解SpringBoot如何實(shí)現(xiàn)一個(gè)實(shí)時(shí)更新的進(jìn)度條

結(jié)尾

這次結(jié)合了前端的定時(shí)任務(wù),后臺事務(wù)以及異步,總的來說還是一次&#128581;&#8205;不錯(cuò)的體驗(yàn)。

看完上述內(nèi)容,是不是對詳解SpringBoot如何實(shí)現(xiàn)一個(gè)實(shí)時(shí)更新的進(jìn)度條有進(jìn)一步的了解,如果還想學(xué)習(xí)更多內(nèi)容,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道。

網(wǎng)站名稱:詳解SpringBoot如何實(shí)現(xiàn)一個(gè)實(shí)時(shí)更新的進(jìn)度條
當(dāng)前地址:http://m.kartarina.com/article18/gogcdp.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供定制網(wǎng)站品牌網(wǎng)站設(shè)計(jì)App設(shè)計(jì)網(wǎng)站維護(hù)網(wǎng)站導(dǎo)航Google

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)

成都定制網(wǎng)站建設(shè)
主站蜘蛛池模板: 99久久人妻无码精品系列| 在线观看免费无码视频| 国产成人无码精品久久二区三区| 无码精品A∨在线观看无广告| 亚洲日韩VA无码中文字幕| 人妻无码久久一区二区三区免费| 国产精品无码免费播放| 久久久久琪琪去精品色无码| 国产一区二区三区无码免费| 无码人妻丰满熟妇区免费| 免费A级毛片无码A| 99热门精品一区二区三区无码| 国产真人无码作爱免费视频| 国产精品无码一区二区三区不卡| 久久午夜无码鲁丝片直播午夜精品| 国产乱人伦中文无无码视频试看| 日韩人妻无码精品一专区| 亚洲自偷自偷偷色无码中文| 无码AV动漫精品一区二区免费| 亚洲av中文无码字幕色不卡 | 亚洲av无码专区国产乱码在线观看| 亚洲精品天堂无码中文字幕| 亚洲av无码成人黄网站在线观看| 国产综合无码一区二区色蜜蜜| 亚洲av永久无码精品网址| 无码超乳爆乳中文字幕久久| 国产午夜精品无码| 亚洲AV无码国产精品色午友在线| 日韩a级无码免费视频| 中文无码乱人伦中文视频在线V| 精品无码成人网站久久久久久| 亚洲中文字幕无码久久2020 | 亚洲国产精品成人精品无码区在线 | 中文字幕无码精品亚洲资源网久久| 国产精品无码无片在线观看| 国产日韩精品中文字无码| 亚洲精品无码鲁网中文电影| 国产亚洲AV无码AV男人的天堂| 国产AV无码专区亚洲A∨毛片| 国产成人无码一区二区三区| 日日麻批免费40分钟无码|