Asp.Net Core 中的“虛擬目錄”

来源:https://www.cnblogs.com/EminemJK/archive/2020/07/23/13362368.html
-Advertisement-
Play Games

寫在前面 現在部署Asp.Net Core應用已經不再限制於Windows的IIS上,更多的是Docker容器、各種反向代理來部署。也有少部分用IIS部署的,IIS部署確實是又快又簡單,圖形化操作三下五除二就可以發佈好一個系統了。在過去Asp.Net MVC 項目部署的時候,還常常使用IIS一個功能 ...


寫在前面

  現在部署Asp.Net Core應用已經不再限制於Windows的IIS上,更多的是Docker容器、各種反向代理來部署。也有少部分用IIS部署的,IIS部署確實是又快又簡單,圖形化操作三下五除二就可以發佈好一個系統了。在過去Asp.Net MVC 項目部署的時候,還常常使用IIS一個功能——虛擬目錄

虛擬目錄可以直接定位到非項目的其他路徑,將路徑作為網站的一部分,可實現上傳文件保存到其他盤符或間接的使用項目以外的靜態文件。在Asp.Net MVC中從虛擬路徑中存取文件也很簡單,如 Server.MapPath("~/Upload/liohuang.jpg"); 

但在Asp.Net Core上不同,它被抽象出一個“文件系統”,也就是FileProvider。FileProvider是對所有實現了IFileProvider介面的所有類型以及對應對象的統稱,在Artech蔣老師的《.NET Core的文件系統[2]:FileProvider是個什麼東西?》文章中已經透析了,這裡不在羅里吧嗦了。

這篇文章要解決的內容是:Asp.Net Core應用中,如何優雅的使用“虛擬目錄”。

實操

  首先,新建一個.Net Core WebApi空項目部署在D盤,“虛擬目錄”假設物理路徑在F盤,分別創建三個測試目錄: F:/test1 、 F:/test2 和 F:/test3 ,目錄里分別存放對應的文件 1/2/3.jpg 和 mybook.txt 。

讀取虛擬目錄文件

  在 Startup.ConfigureServices 註入 IFileProvider :

   services.AddSingleton<IFileProvider>(new PhysicalFileProvider("F:\\test1"));

新建一個控制器,讀取 mybook.txt 中的內容:

    [ApiController]
    [Route("[controller]/[action]")]
    public class LioHuangController : ControllerBase
    {
        [HttpGet]
        public object GetFiles([FromServices]IFileProvider fileProvider)
        {
            var file = fileProvider.GetFileInfo("mybook.txt");
            if (file.Exists)
            {
                return ReadTxtContent(file.PhysicalPath);
            }
            return 0;
        }

        /// <summary>
        /// 讀取文本 (原文地址:https://www.cnblogs.com/EminemJK/p/13362368.html/// </summary>
        private string ReadTxtContent(string Path)
        {
            if (!System.IO.File.Exists(Path))
            {
                return "Not found!";
            }
            using (StreamReader sr = new StreamReader(Path, Encoding.UTF8))
            {
                StringBuilder sb = new StringBuilder();
                string content;
                while ((content = sr.ReadLine()) != null)
                {
                    sb.Append(content);
                }
                return sb.ToString();
            }
        }
    }

訪問介面,介面讀取文件之後,返回內容:

 IFileProvider 介面採用目錄來組織文件,並統一使用 IFileInfo 介面來表示, PhysicalPath 表示文件的物理路徑。

    public interface IFileInfo
    {
        bool Exists { get; }
        bool IsDirectory { get; }
        DateTimeOffset LastModified { get; }
        string Name { get; }
        string PhysicalPath { get; }
        Stream CreateReadStream();
    }

 如多個虛擬目錄,怎麼處理?簡單,註入多個 IFileProvider 即可,

   services.AddSingleton<IFileProvider>(new PhysicalFileProvider("F:\\test1"));
   services.AddSingleton<IFileProvider>(new PhysicalFileProvider("F:\\test2"));
   services.AddSingleton<IFileProvider>(new PhysicalFileProvider("F:\\test3"));    

代碼修改為:

   public object GetFiles([FromServices] IEnumerable<IFileProvider> fileProviders)

 IEnumerable<IFileProvider> fileProviders 介面數組將會有三個,按註入的順序對應不同的目錄。當然,註入 IFileProvider 的時候,就可以封裝一層了,下麵再講。

另外,有的說直接 ReadTxtContent("F:\test1\mybook.txt"); 不香嗎?香,Asp.Net Core的訪問許可權要比Asp.Net MVC之前老版本項目要高許多,確實是可以直接讀取項目以外的文件,但是並不適合直接去訪問,除非說你只有一個地方使用到,那麼就可以直接讀取,但靜態的文件的訪問,就訪問不到了,僅僅是後臺讀取而已。所以統一使用 IFileProvider 來約束,代碼的可維護性要高許多。

靜態文件訪問

  在Startup.Configure設置靜態文件目錄,即可:

      app.UseStaticFiles(new StaticFileOptions()
            {
                FileProvider = new PhysicalFileProvider("F:\\test1"),
                RequestPath = "/test"
            });;
      app.UseStaticFiles(new StaticFileOptions()
            {
                FileProvider = new PhysicalFileProvider("F:\\test2"),
                RequestPath = "/test"
            });
      app.UseStaticFiles(new StaticFileOptions()
            {
                FileProvider = new PhysicalFileProvider("F:\\test3"),
                RequestPath = "/test"
            });

 FileProvider 同上面所說的,設置好物理路徑的根目錄, RequestPath 則是訪問路徑的首碼,必須是斜桿 “/” 開頭,訪問地址首碼則為: https://localhost:5001/test/ 。設置好之後,就可以訪問項目以外的路徑了。

如在IIS部署的時候 ,可以直接忽略IIS中的虛擬目錄設置,完完全全可以通過註入的配置來設置達到“虛擬目錄”的效果。

簡化配置

  為了方便達到真實項目中可以直接使用,那麼就要設置為可配置的:

在 appsettings.json 中設置:

{
  "Logging": {
    "LogLevel": {
      "Default": "Information",
      "Microsoft": "Warning",
      "Microsoft.Hosting.Lifetime": "Information"
    }
  },
  "AllowedHosts": "*",

  "VirtualPath": [
    {
      "RealPath": "F:\\test1", //真實路徑
      "RequestPath": "/test",
      "Alias": "first"
    },
    {
      "RealPath": "F:\\test2", //真實路徑
      "RequestPath": "/test",
      "Alias": "second"
    },
    {
      "RealPath": "F:\\test3", //真實路徑
      "RequestPath": "/test",
      "Alias": "third"
    }
  ]
}

創建對應的實體映射:

    public class VirtualPathConfig
    {
        public List<PathContent> VirtualPath { get; set; }
    }

    public class PathContent
    {
        public string RealPath { get; set; }

        public string RequestPath { get; set; }

        public string Alias { get; set; }
    }

在 PhysicalFileProvider 上封裝一層,加入別名便於獲取:

    public class MyFileProvider : PhysicalFileProvider
    {
        public MyFileProvider(string root, string alias) : base(root)
        {
            this.Alias = alias;
        }

        public MyFileProvider(string root, Microsoft.Extensions.FileProviders.Physical.ExclusionFilters filters, string alias) : base(root, filters)
        {
            this.Alias = alias;
        }

        /// <summary>
        /// 別名
        /// </summary>
        public string Alias { get; set; }
    }

調整 Startup.ConfigureServices 和 Startup.Configure :

        public void ConfigureServices(IServiceCollection services)
        {
            services.AddControllers();
            services.Configure<VirtualPathConfig>(Configuration);

            var config = Configuration.Get<VirtualPathConfig>().VirtualPath;
            config.ForEach(f => 
            {
                services.AddSingleton(new MyFileProvider(f.RealPath,f.Alias));
            });
        }

        public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
        {
            if (env.IsDevelopment())
            {
                app.UseDeveloperExceptionPage();
            }

            var config = Configuration.Get<VirtualPathConfig>().VirtualPath;
            config.ForEach(f =>
            {
                app.UseStaticFiles(new StaticFileOptions()
                {
                    FileProvider = new PhysicalFileProvider(f.RealPath),
                    RequestPath =f.RequestPath
                });
            });
             
            app.UseRouting();
            app.UseAuthorization();
            app.UseEndpoints(endpoints =>
            {
                endpoints.MapControllers();
            });
        }

最後,調整調用方式,即可。

        [HttpGet]
        public object GetFiles([FromServices] IEnumerable<MyFileProvider> fileProviders)
        {
            var file = fileProviders.FirstOrDefault(x=>x.Alias=="first").GetFileInfo("mybook.txt");
            if (file.Exists)
            {
                return ReadTxtContent(file.PhysicalPath);
            }
            return 0;
        }

最後

  物理文件系統的抽象通過 PhysicalFileProvider 這個 FileProvider 來實現,藉助 IFileProvider 的特點,其實可以擴展實現輕量“雲盤”的功能了,而不僅僅只是實現IIS虛擬目錄功能。搞定,今晚不加班!


 本文同步在DotNetGeek(ID:dotNetGeek)公眾號發佈


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

-Advertisement-
Play Games
更多相關文章
  • 例如導出指定文件夾中的所有mp4文件的文件名 1 @echo off 2 for /r C:\Users\Administrator\Desktop\video %%i in (*.mp4) do ( 3 echo %%i 4 echo %%i>>C:\Users\Administrator\Des ...
  • 看到一篇關於Java學習體系的總結,在這裡給大家分享一下: 一、Java SE部分 java基礎:基礎語法;面向對象(重點);集合框架(重點);常見類庫API;java界面編程:AWT;事件機制;Swing;java高級知識:Annotation;IO和NIO、AIO;多線程、線程池;阻塞、非阻塞、 ...
  • 介面概述: 1、介面是Java語言中的一種引用類型,是方法的"集合",所以介面的內部主要就是定義方法,包含常量,抽象方法(JDK 7及以前),額外增加預設方法和靜態方法(JDK 8),額外增加私有方法(jdk9)。 介面的定義,它與定義類方式相似,但是使用 interface 關鍵字。它也會被編譯成 ...
  • 智能指針可以對動態資源進行管理,保證任何情況下,已經構造的對象能夠安全的自動銷毀。 ...
  • 任何Java開發人員的必備技能 現在,讓我們一起探索這個Java開發人員路線圖,並找出應該學習哪些工具,框架,庫和API。 1.工具 路線圖工具部分分為兩個主要部分。首先,IDE是您的主要工具,它幾乎可以完成您所要求的一切,例如編譯,運行,調試,分析,測試,比較文件和代碼,重構等等。 第二部分是構建 ...
  • 內容簡介 · · · · · · 本書是經典教程的全新改版,作者根據Python 3.0版本的種種變化,全面改寫了書中內容,做到既能“瞻前”也能“顧後”。本書層次鮮明、結構嚴謹、內容翔實,特別是在最後幾章,作者將前面講述的內容應用到了10個引人入勝的項目中,並以模板的形式介紹了項目的開發過程。本書既 ...
  • 方法一:新增models的Manager方法 下麵就直接發代碼了 class RandomManager(models.Manager): def get_queryset(self): return super(RandomManager, self).get_queryset().order_b ...
  • BitMap的基本思想就是用一個bit位來標記某個元素對應的Value,而Key即是該元素。由於採用了Bit為單位來存儲數據,因此可以大大節省存儲空間。 BitMap可以看成一種數據結構。 ...
一周排行
    -Advertisement-
    Play Games
  • 前言 微服務架構已經成為搭建高效、可擴展系統的關鍵技術之一,然而,現有許多微服務框架往往過於複雜,使得我們普通開發者難以快速上手並體驗到微服務帶了的便利。為瞭解決這一問題,於是作者精心打造了一款最接地氣的 .NET 微服務框架,幫助我們輕鬆構建和管理微服務應用。 本框架不僅支持 Consul 服務註 ...
  • 先看一下效果吧: 如果不會寫動畫或者懶得寫動畫,就直接交給Blend來做吧; 其實Blend操作起來很簡單,有點類似於在操作PS,我們只需要設置關鍵幀,滑鼠點來點去就可以了,Blend會自動幫我們生成我們想要的動畫效果. 第一步:要創建一個空的WPF項目 第二步:右鍵我們的項目,在最下方有一個,在B ...
  • Prism:框架介紹與安裝 什麼是Prism? Prism是一個用於在 WPF、Xamarin Form、Uno 平臺和 WinUI 中構建鬆散耦合、可維護和可測試的 XAML 應用程式框架 Github https://github.com/PrismLibrary/Prism NuGet htt ...
  • 在WPF中,屏幕上的所有內容,都是通過畫筆(Brush)畫上去的。如按鈕的背景色,邊框,文本框的前景和形狀填充。藉助畫筆,可以繪製頁面上的所有UI對象。不同畫筆具有不同類型的輸出( 如:某些畫筆使用純色繪製區域,其他畫筆使用漸變、圖案、圖像或繪圖)。 ...
  • 前言 嗨,大家好!推薦一個基於 .NET 8 的高併發微服務電商系統,涵蓋了商品、訂單、會員、服務、財務等50多種實用功能。 項目不僅使用了 .NET 8 的最新特性,還集成了AutoFac、DotLiquid、HangFire、Nlog、Jwt、LayUIAdmin、SqlSugar、MySQL、 ...
  • 本文主要介紹攝像頭(相機)如何採集數據,用於類似攝像頭本地顯示軟體,以及流媒體數據傳輸場景如傳屏、視訊會議等。 攝像頭採集有多種方案,如AForge.NET、WPFMediaKit、OpenCvSharp、EmguCv、DirectShow.NET、MediaCaptre(UWP),網上一些文章以及 ...
  • 前言 Seal-Report 是一款.NET 開源報表工具,擁有 1.4K Star。它提供了一個完整的框架,使用 C# 編寫,最新的版本採用的是 .NET 8.0 。 它能夠高效地從各種資料庫或 NoSQL 數據源生成日常報表,並支持執行複雜的報表任務。 其簡單易用的安裝過程和直觀的設計界面,我們 ...
  • 背景需求: 系統需要對接到XXX官方的API,但因此官方對接以及管理都十分嚴格。而本人部門的系統中包含諸多子系統,系統間為了穩定,程式間多數固定Token+特殊驗證進行調用,且後期還要提供給其他兄弟部門系統共同調用。 原則上:每套系統都必須單獨接入到官方,但官方的接入複雜,還要官方指定機構認證的證書 ...
  • 本文介紹下電腦設備關機的情況下如何通過網路喚醒設備,之前電源S狀態 電腦Power電源狀態- 唐宋元明清2188 - 博客園 (cnblogs.com) 有介紹過遠程喚醒設備,後面這倆天瞭解多了點所以單獨加個隨筆 設備關機的情況下,使用網路喚醒的前提條件: 1. 被喚醒設備需要支持這WakeOnL ...
  • 前言 大家好,推薦一個.NET 8.0 為核心,結合前端 Vue 框架,實現了前後端完全分離的設計理念。它不僅提供了強大的基礎功能支持,如許可權管理、代碼生成器等,還通過採用主流技術和最佳實踐,顯著降低了開發難度,加快了項目交付速度。 如果你需要一個高效的開發解決方案,本框架能幫助大家輕鬆應對挑戰,實 ...