.Net Core WebAPI + Axios +Vue 實現下載與下載進度條

来源:https://www.cnblogs.com/EminemJK/archive/2020/06/09/13074053.html
-Advertisement-
Play Games

故事的開始 老闆說:系統很慢,下載半個小時無法下載,是否考慮先壓縮再給用戶下載? 本來是已經壓縮過了,不過第一反應應該是用戶下的數量多,導致壓縮包很大,然後自己測試發現,只是等待的時間比較久而已,仍然是下載狀態中,並不是系統慢,但是用戶體驗肯定是最直觀的,確實是我們做得不夠好,單純彈出遮罩層顯示冰冷 ...


故事的開始

老闆說:系統很慢,下載半個小時無法下載,是否考慮先壓縮再給用戶下載?

                     

 

  本來是已經壓縮過了,不過第一反應應該是用戶下的數量多,導致壓縮包很大,然後自己測試發現,只是等待的時間比較久而已,仍然是下載狀態中,並不是系統慢,但是用戶體驗肯定是最直觀的,確實是我們做得不夠好,單純彈出遮罩層顯示冰冷的“拼命載入中……”,對用戶來說確實不夠友好。嗯,瞭解實際情況了,那就開擼,增加用戶體驗。

 解決它

效果圖:

 

 

Vue+ElementUI

 <el-progress v-if="dlProgress>0" :text-inside="true" :stroke-width="18" :percentage="dlProgress" status="success" style="margin-bottom:10px"></el-progress>

Axios

downloadTask(index,row) {
      let own =this;
      this.fullscreenLoading = true;
      this.axios({
        method: 'post',
        url: this.baseUrl + '/api/Task/DownLoad',
        data: {id: row.id},
        responseType: 'blob',
     //敲黑板 onDownloadProgress (progress) { own.dlProgress
=Math.round(progress.loaded / progress.total * 100); } }) .then((res) => { this.fullscreenLoading = false; let fileName = decodeURI(res.headers["content-disposition"].split("=")[1]); let url = window.URL.createObjectURL(new Blob([res.data])); let link = document.createElement('a'); link.style.display = 'none'; link.href = url; link.setAttribute('download', fileName); document.body.appendChild(link) link.click() this.$message.success('下載成功'); }) .catch(() => { this.fullscreenLoading = false; }); },

下載:

    public static class HttpContextExtension
    {
        /// <summary>
        /// 獲取客戶Ip
        /// </summary>
        /// <param name="context"></param>
        /// <returns></returns>
        public static string GetClientUserIp(this HttpContext context)
        {
            var ip = "";
            if (context.Request.Headers.ContainsKey("X-Forwarded-For"))
            {
                ip = context.Request.Headers["X-Forwarded-For"].ToString();
            }
            if (string.IsNullOrEmpty(ip) && context.Request.Headers.ContainsKey("X-Real-IP"))
            {
                ip = context.Request.Headers["X-Real-IP"].ToString();
            }
            if (string.IsNullOrEmpty(ip))
            {
                ip = context.Connection.RemoteIpAddress.MapToIPv4().ToString();
            }
            if (string.IsNullOrEmpty(ip))
                return ip;
            var sp = ip.RemoveSpacing().Split(',');
            return sp[0];
        }

        /// <summary>
        /// 通過文件流下載文件
        /// </summary>
        /// <param name="context"></param>
        /// <param name="filePath">文件完整路徑</param>
        /// <param name="contentType">訪問這裡 https://tool.oschina.net/commons </param>
        public static void DownLoadFile(this HttpContext context,string filePath, string contentType= "application/octet-stream")
        {
            var fileName = Path.GetFileName(filePath);

            int bufferSize = 1024; 
            context.Response.ContentType = contentType;
            context.Response.Headers.Append("Content-Disposition", "attachment;filename=" + HttpUtility.UrlEncode(fileName));
            context.Response.Headers.Append("Charset", "utf-8");
            context.Response.Headers.Append("Access-Control-Expose-Headers", "Content-Disposition");
            //context.Response.Headers.Append("Access-Control-Allow-Origin", "*");
            //使用FileStream開始迴圈讀取要下載文件的內容
            using (FileStream fs = new FileStream(filePath, FileMode.Open, FileAccess.Read))
            {
                using (context.Response.Body) 
                {
                    long contentLength = fs.Length; 
                    context.Response.ContentLength = contentLength;

                    byte[] buffer;
                    long hasRead = 0;  
                    while (hasRead < contentLength)
                    {
                        if (context.RequestAborted.IsCancellationRequested)
                        {
                            break;
                        }

                        buffer = new byte[bufferSize];
                        //從下載文件中讀取bufferSize(1024位元組)大小的內容到伺服器記憶體中
                        int currentRead = fs.Read(buffer, 0, bufferSize);
                        context.Response.Body.Write(buffer, 0, currentRead);
                        context.Response.Body.Flush();
                        hasRead += currentRead;
                    }
                    context.Response.Body.Close();
                }
                fs.Close();
            }
        }

        /// <summary>
        /// 通過文件流下載文件
        /// </summary>
        /// <param name="context"></param>
        /// <param name="filePath">文件完整路徑</param>
        /// <param name="contentType">訪問這裡 https://tool.oschina.net/commons </param>
        public static void DownLoadFile(this HttpContext context,string fileName, byte[] fileByte, string contentType = "application/octet-stream")
        {
            int bufferSize = 1024;
            
            context.Response.ContentType = contentType;
            context.Response.Headers.Append("Content-Disposition", "attachment;filename=" + HttpUtility.UrlEncode(fileName));
            context.Response.Headers.Append("Charset", "utf-8");
            context.Response.Headers.Append("Access-Control-Expose-Headers", "Content-Disposition");
         
            //context.Response.Headers.Append("Access-Control-Allow-Origin", "*");
            //使用FileStream開始迴圈讀取要下載文件的內容
            using (Stream fs = new MemoryStream(fileByte))
            {
                using (context.Response.Body)
                {
                    long contentLength = fs.Length;
                    context.Response.ContentLength = contentLength;

                    byte[] buffer;
                    long hasRead = 0;
                    while (hasRead < contentLength)
                    {
                        if (context.RequestAborted.IsCancellationRequested)
                        {
                            break;
                        }
                        
                        buffer = new byte[bufferSize];
                        //從下載文件中讀取bufferSize(1024位元組)大小的內容到伺服器記憶體中
                        int currentRead = fs.Read(buffer, 0, bufferSize);
                        context.Response.Body.Write(buffer, 0, currentRead);
                        context.Response.Body.Flush();
                        hasRead += currentRead;
                    }
                }
            }
        }
    }
View Code

完美~

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 作者:美圖博客 https://www.meitubk.com/zatan/386.html 前言 最近突然有個奇妙的想法,就是當我對著電腦屏幕的時候,電腦會先識別屏幕上的人臉是否是本人,如果識別是本人的話需要回答電腦說的暗語,答對了才會解鎖並且有三次機會。如果都沒答對就會發送郵件給我,通知有人在動 ...
  • 前言 已經好多年都沒有使用 SVN 了,它的一些使用套路現在也忘記得差不多了,最近由於手賤給其中一個分支加鎖了,等解鎖的時候忘記怎麼解開了,類似下麵的樣子: 自己挖的坑還是自己來填吧,這裡簡單記錄一下。 解決方案 如果項目加鎖了,我們可以查看到當前鎖的狀態,如下圖所示: 如果想對單個文件解鎖,只需要 ...
  • Volo.Abp.MailKit封裝繼承MailKit庫,為Abp郵件發送提供了快捷實現。 郵箱配置 qq郵箱支持smtp功能,需要去申請開通。參考qq郵箱設置,最重要的是smtp發送郵件,qq郵箱對應的密碼不是用戶的qq郵箱密碼,而是需要申請生成的授權碼。 在項目的appsettings.json ...
  • 什麼是網關 簡單點說網關是一個Api伺服器,是系統的唯一入口。為每個客戶端提供一個定製的Restful API。同時它還需要具有一些業務之外的責任:鑒權。靜態響應等處理。 為什麼需要gateway 我們知道我們要進入一個服務本身,並不是一件容易的事情。服務本身有自己的通訊協議,這種協議往往不能很好的 ...
  • 網上很多創建Widows Service 的方法,其安裝和運行服務的方式大多是通過cmd 命令來實現的,這裡將通過控制台的形式實現安裝、運行和卸載服務。 主頁代碼: 1 using KQService.Job; 2 using KQService.Utils; 3 using System; 4 u ...
  • 這是一文說通系列的第二篇,裡面有些內容會用到第一篇中間件的部分概念。如果需要,可以參看第一篇:一文說通Dotnet Core的中間件 一、前言 後臺任務在一些特殊的應用場合,有相當的需求。 比方,我們需要實現一個定時任務、或周期性的任務、或非API輸出的業務響應、或不允許併發的業務處理,像提現、支付 ...
  • 系列文章 基於 abp vNext 和 .NET Core 開發博客項目 - 使用 abp cli 搭建項目 基於 abp vNext 和 .NET Core 開發博客項目 - 給項目瘦身,讓它跑起來 基於 abp vNext 和 .NET Core 開發博客項目 - 完善與美化,Swagger登場 ...
  • 動手造輪子:實現一個簡單的依賴註入(三) 支持屬性註入 Intro 前面寫了幾篇依賴註入的文章,有興趣的小伙伴可以參考文末 Reference 部分中的鏈接,一直有小伙伴希望增加屬性註入的支持,昨天試著加了一下,思路很簡單,在獲取到服務實例之後檢查實例中有沒有需要註入的屬性,如果有並且不為 null ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...