溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Asp.Net Core 中實現(xiàn)“虛擬目錄”的方法

發(fā)布時間:2020-08-15 09:32:44 來源:億速云 閱讀:516 作者:小新 欄目:開發(fā)技術(shù)

這篇文章給大家分享的是有關(guān)Asp.Net Core 中實現(xiàn)“虛擬目錄”的方法的內(nèi)容。小編覺得挺實用的,因此分享給大家做個參考。一起跟隨小編過來看看吧。

寫在前面

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

Asp.Net Core 中實現(xiàn)“虛擬目錄”的方法

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

但在Asp.Net Core上不同,它被抽象出一個“文件系統(tǒng)”,也就是FileProvider。FileProvider是對所有實現(xiàn)了IFileProvider接口的所有類型以及對應(yīng)對象的統(tǒng)稱,在Artech蔣老師的《.NET Core的文件系統(tǒng)[2]:FileProvider是個什么東西?》文章中已經(jīng)透析了,這里不在羅里吧嗦了。

這篇文章要解決的內(nèi)容是:Asp.Net Core應(yīng)用中,如何優(yōu)雅的使用“虛擬目錄”。

實操

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

讀取虛擬目錄文件

  在Startup.ConfigureServices注入IFileProvider:

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

新建一個控制器,讀取mybook.txt中的內(nèi)容:

[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();
   }
  }
 }

訪問接口,接口讀取文件之后,返回內(nèi)容:

Asp.Net Core 中實現(xiàn)“虛擬目錄”的方法

IFileProvider接口采用目錄來組織文件,并統(tǒng)一使用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接口數(shù)組將會有三個,按注入的順序?qū)?yīng)不同的目錄。當(dāng)然,注入IFileProvider的時候,就可以封裝一層了,下面再講。

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

靜態(tài)文件訪問

  在Startup.Configure設(shè)置靜態(tài)文件目錄,即可:

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同上面所說的,設(shè)置好物理路徑的根目錄,RequestPath則是訪問路徑的前綴,必須是斜桿“/”開頭,訪問地址前綴則為:https://localhost:5001/test/。設(shè)置好之后,就可以訪問項目以外的路徑了。

Asp.Net Core 中實現(xiàn)“虛擬目錄”的方法

Asp.Net Core 中實現(xiàn)“虛擬目錄”的方法

Asp.Net Core 中實現(xiàn)“虛擬目錄”的方法

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

簡化配置

  為了方便達(dá)到真實項目中可以直接使用,那么就要設(shè)置為可配置的:

在appsettings.json中設(shè)置:

{
 "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"
 }
 ]
}

創(chuàng)建對應(yīng)的實體映射:

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; }
  }

調(diào)整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();
      });
    }

最后,調(diào)整調(diào)用方式,即可。

[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;
    }

  物理文件系統(tǒng)的抽象通過PhysicalFileProvider這個FileProvider來實現(xiàn),借助IFileProvider的特點,其實可以擴展實現(xiàn)輕量“云盤”的功能了,而不僅僅只是實現(xiàn)IIS虛擬目錄功能。

感謝各位的閱讀!關(guān)于Asp.Net Core 中實現(xiàn)“虛擬目錄”的方法就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,讓大家可以學(xué)到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到吧!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI