java實戰(zhàn)--異步Excel下載

1瘪松、說明

實際業(yè)務中咸作,系統(tǒng)會涉及Excel報表下載共功能,當業(yè)務數(shù)據(jù)體量過大宵睦,為避免用戶點擊下載后長時間等待记罚,可設計成異步Excel(生成Excel下載Excel兩步)下載,用戶點擊下載任務觸達后壳嚎,可進行其他業(yè)務操作桐智,稍后在業(yè)務報表模塊中查看生成的Excel信息,在點擊已生成完成的的Excel文件鏈接下載即可诬辈。

2、 設計方案

2.1 業(yè)務交互流程

(1)用戶點擊業(yè)務下載按鈕

image.png

(2)系統(tǒng)收到下載請求荐吉,后臺異步去生成Excel文件焙糟,彈框反饋用戶下載請求已觸達,稍后去報表模塊查看報表
image.png

(3)報表中心看到后臺生成好的Excel報表名稱及時間
image.png

2.2 設計示意圖

image.png

【說明】
a.用戶點擊下載Excel
b.異步生成Excel文件样屠,文件上傳到文件存儲OSS服務器穿撮,返回文件地址
c.文件名稱及地址保存數(shù)據(jù)庫
d.用戶在報表中心展示文件地址超鏈接缺脉。下載從OSS云服務器下載(可CDN加速)

3、實現(xiàn)

3.1 數(shù)據(jù)庫表設計

  • 導出文件表point_export_file
CREATE TABLE `point_export_file` (
  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT '主鍵',
  `zone_id` varchar(32) NOT NULL DEFAULT '0' COMMENT '導出文件所屬區(qū)部',
  `type` tinyint(4) NOT NULL DEFAULT '0' COMMENT '文件類型',
  `file_name` varchar(256) DEFAULT '' COMMENT '文件名稱',
  `file_url` varchar(256) DEFAULT '' COMMENT '文件url',
  `operator_id` bigint(20) DEFAULT NULL COMMENT '操作人id',
  `operator_name` varchar(256) NOT NULL DEFAULT 'admin' COMMENT '用戶名稱',
  `down_count` smallint(6) unsigned NOT NULL DEFAULT '0' COMMENT '下載次數(shù)',
  `created_at` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '創(chuàng)建時間',
  `updated_at` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '最近更新時間',
  PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='導出文件列表';

3.2 程序代碼實現(xiàn)

(1)controller類悦穿,接收業(yè)務導出請求攻礼,觸發(fā)異步導出任務。

@Qualifier("excelThreadPool")
@Autowired
private ExecutorService executorService;

@ApiOperation("導出積分明細")
@GetMapping(value = "/api/point/export")
public Response<Boolean> exportConsumeRecord(PointExportCriteria criteria) {
     // 觸發(fā)異步事件
     Future<?> future = executorService.submit( () -> {
          final String fileUrl = pointService.getPointRecordFileUrl(criteria);} );
     return Response.ok(true);
}

(2)配置類config栗柒,配置異步線程池礁扮。

@Bean("excelThreadPool")
   public ExecutorService buildExcelThreadPool() {
       int cpuNum = Runtime.getRuntime().availableProcessors();
       BlockingQueue<Runnable> workQueue = new LinkedBlockingQueue<>(1000);
       ThreadFactory threadFactory = new ThreadFactoryBuilder().setNameFormat("excel-pool-%d").build();
       return new ThreadPoolExecutor( cpuNum * 2 + 1, 5 * cpuNum,
               2, TimeUnit.MINUTES, workQueue, threadFactory);
   }

(3)核心異步業(yè)務實現(xiàn)類,讀取業(yè)務數(shù)據(jù)生成Excel文件瞬沦,并觸發(fā)上傳文件太伊。

public String getPointRecordFileUrl(PointExportCriteria criteria){
      // todo 1查詢數(shù)據(jù)
      List<Point>  pointList  = new ArrayList();
       int pageNo = 1;
       criteria.setPageNo(pageNo);
       criteria.setPageSize(PAGE_SIZE);
       String fileName = "自定義文件名頭-" + DateTime.now().toString("yyyyMMddHHmmss") + ".xlsx";
       String filePath = FILE_PATH +fileName;
       int total = PAGE_SIZE;
        // todo 2.使用alibaba Excel去寫文檔
       ExcelWriter excelWriter = null;
       try {
            // 這里 需要指定寫用哪個class去寫
            excelWriter = EasyExcel.write(filePath, Point.class).build();
            // 這里注意 如果同一個sheet只要創(chuàng)建一次
            WriteSheet writeSheet = EasyExcel.writerSheet("電子券使用記錄").build();
            while (true) {
                criteria.setPageNo(pageNo);
                Paging<Point> paging = point.pagingMapper(criteria);

                if (paging.getData().isEmpty()) {
                    log.warn("this paging now do not has coupon consume record,criteria {},pageNo {}", criteria, pageNo);
                    break;
                }
                List<Point> pointList = paging.getData();
                total= pointList.size();
                exportScopeDtoList.addAll(pointList);
                excelWriter.write(exportScopeDtoList, writeSheet);
                exportScopeDtoList.clear();
                pageNo++;
                if (total < PAGE_SIZE) {
                    break;
                }
            }

        } catch (Exception e) {
            log.error("export failed,cause:{} ", Throwables.getStackTraceAsString(e));
            throw new JsonResponseException(500,"point.export.fail");
        } finally {
            // 千萬別忘記finish 會幫忙關(guān)閉流
            if (excelWriter != null) {
                excelWriter.finish();
            }
        }
        // 獲取下載鏈接
        String fileUrl = getFileUrl(filePath, fileName, currentUser,zoneIds);
        log.info("導出積分 done criteria: {},cost: {} ms",criteria,stopwatch.elapsed(TimeUnit.MILLISECONDS));
        return fileUrl;
}

(4)文件上傳,OSS文件存儲逛钻,返回文件地址僚焦,并保存到導出文件列庫表。

    public String getFileUrl(String filePath, String fileName, BaseUser currentUser, List<String> zoneIds) {
        File file = new File(filePath);
        // 上傳云服務器
        String fileUrl = ossBlobClient.doUpload(file);
        PointExportFile exportFile = new PointExportFile();
        exportFile.setFileUrl(fileUrl);
        exportFile.setType(0);
        exportFile.setFileName(fileName);
        exportFile.setOperatorId(currentUser.getId());
        exportFile.setOperatorName(currentUser.getName());
        if (!CollectionUtils.isEmpty(zoneIds)) {
            exportFile.setZoneId(zoneIds.get(0));
        }
        exportFile.setDownCount(0);
        Response<Long> longResponse = pointExportFileMapper.create(exportFile);
        if (!longResponse.isSuccess()) {
            log.error("create export file record fail,cause {}",longResponse.getError());
            throw new JsonResponseException(longResponse.getError());
        }
        return fileUrl;
    }

(5)文件列表查詢曙痘,返回文件鏈接地址芳悲,名稱信息。前端通過Excel鏈接去完成下載操作边坤。

    @ApiOperation("文件導出分頁查詢")
    @RequestMapping(value = "/export/paging", method = RequestMethod.GET)
      public Response<Paging<PointExportFile>>  paging(
            @RequestParam(required = true,defaultValue = "1")Integer pageNo,
            @RequestParam(required = true,defaultValue = "10")Integer pageSize){
        if (log.isDebugEnabled()) {
            log.debug("start paging export file ,pageNo: {},pageSize: {}", pageNo,pageSize);
        }
        ExportFileCriteria exportFileCriteria = new ExportFileCriteria();
        exportFileCriteria.setPageNo(pageNo);
        exportFileCriteria.setPageSize(pageSize);
        exportFileCriteria.setUserZoneIds(ManageZoneUtil.queryUserManageZone());
        Response<Paging<ExportFile>> resp = exportFileReadService.paging(exportFileCriteria);
        if(!resp.isSuccess()){
            log.error("failed to paging export file cause: {}",
                    resp.getError());
            throw new JsonResponseException(resp.getError());
        }
        return resp;
    }
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末名扛,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子惩嘉,更是在濱河造成了極大的恐慌罢洲,老刑警劉巖,帶你破解...
    沈念sama閱讀 219,188評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件文黎,死亡現(xiàn)場離奇詭異惹苗,居然都是意外死亡,警方通過查閱死者的電腦和手機耸峭,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,464評論 3 395
  • 文/潘曉璐 我一進店門桩蓉,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人劳闹,你說我怎么就攤上這事院究。” “怎么了本涕?”我有些...
    開封第一講書人閱讀 165,562評論 0 356
  • 文/不壞的土叔 我叫張陵业汰,是天一觀的道長。 經(jīng)常有香客問我菩颖,道長样漆,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,893評論 1 295
  • 正文 為了忘掉前任晦闰,我火速辦了婚禮放祟,結(jié)果婚禮上鳍怨,老公的妹妹穿的比我還像新娘。我一直安慰自己跪妥,他們只是感情好鞋喇,可當我...
    茶點故事閱讀 67,917評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著眉撵,像睡著了一般侦香。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上执桌,一...
    開封第一講書人閱讀 51,708評論 1 305
  • 那天鄙皇,我揣著相機與錄音,去河邊找鬼仰挣。 笑死伴逸,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的膘壶。 我是一名探鬼主播错蝴,決...
    沈念sama閱讀 40,430評論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼颓芭!你這毒婦竟也來了顷锰?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,342評論 0 276
  • 序言:老撾萬榮一對情侶失蹤亡问,失蹤者是張志新(化名)和其女友劉穎官紫,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體州藕,經(jīng)...
    沈念sama閱讀 45,801評論 1 317
  • 正文 獨居荒郊野嶺守林人離奇死亡束世,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,976評論 3 337
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了床玻。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片毁涉。...
    茶點故事閱讀 40,115評論 1 351
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖锈死,靈堂內(nèi)的尸體忽然破棺而出贫堰,到底是詐尸還是另有隱情,我是刑警寧澤待牵,帶...
    沈念sama閱讀 35,804評論 5 346
  • 正文 年R本政府宣布其屏,位于F島的核電站,受9級特大地震影響缨该,放射性物質(zhì)發(fā)生泄漏偎行。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,458評論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望睦优。 院中可真熱鬧,春花似錦壮不、人聲如沸汗盘。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,008評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽隐孽。三九已至,卻和暖如春健蕊,著一層夾襖步出監(jiān)牢的瞬間菱阵,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,135評論 1 272
  • 我被黑心中介騙來泰國打工缩功, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留晴及,地道東北人。 一個月前我還...
    沈念sama閱讀 48,365評論 3 373
  • 正文 我出身青樓嫡锌,卻偏偏與公主長得像虑稼,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子势木,可洞房花燭夜當晚...
    茶點故事閱讀 45,055評論 2 355

推薦閱讀更多精彩內(nèi)容