ASP.NET Core File Providers
ASP.NET Core通過對File Providers的使用實現(xiàn)了對文件系統(tǒng)訪問的抽象。
查看或下載示例代碼
File Provider 抽象
File Providers是文件系統(tǒng)之上的一層抽象。它的主要接口是IFileProvider。IFileProvider公開了相應方法用來獲取文件信息(IFileInfo), 目錄信息(IDirectoryContents),以及設置更改通知(通過使用一個IChangeToken)。
IFileInfo接口提供了操作單個文件和目錄的方法和屬性。它有兩個boolean屬性,Exists和IsDirectory,以及兩個描述文件的兩個屬性Name和Length(按字節(jié)),還包括一個LastModified日期屬性。你還可以通過CreateReadStream方法讀取文件內容。
File Provider 實現(xiàn)
有三種對于IFileProvider的實現(xiàn)可供選擇:物理式,嵌入式和復合式。物理式用于訪問實際系統(tǒng)中的文件。嵌入式用于訪問嵌入在程序集中的文件。 復合式則是對前兩種方式的組合使用。
PhysicalFileProvider
PhysicalFileProvider提供了對物理文件系統(tǒng)的訪問。它封裝了System.IO.File類型,范圍限定到一個目錄及其子目錄的所有路徑。這類作用域會限制訪問某個目錄及其子目錄,防止作用域以外的其他操作訪問文件系統(tǒng)。當實例化此類provider時,你必須為它提供一個目錄路徑,以供服務器拿來當做由這個provider發(fā)出的所有請求的基礎路徑(這個provider會限制路徑以外的訪問請求)。在一個ASP.NET Core應用,你可以直接實例化出一個PhysicalFileProvider?provider,或者你也可以通過在控制器和服務中使用構造函數(shù)依賴注入的方式,請求一個IFileProvider接口。后者生成的解決方案通常更靈活以及更便于測試。
要創(chuàng)建一個PhysicalFileProvider其實很簡單,只需要對其實化,再傳遞給它一個物理路徑。之后你就可以通過它的目錄遍歷內容或提供子路徑獲取特定文件的信息。
IFileProvider provider = new PhysicalFileProvider(applicationRoot); IDirectoryContents contents = provider.GetDirectoryContents(""); // the applicationRoot contentsIFileInfo fileInfo = provider.GetFileInfo("wwwroot/js/site.js"); // a file under applicationRoot為了在控制器中請求一個provider,需要在控制器的構造函數(shù)中指定類型參數(shù)并賦值給本地屬性。之后你就可以在你的動作器方法中使用本地實例了。
public class HomeController : Controller{ ? ?private readonly IFileProvider _fileProvider; ? ?public HomeController(IFileProvider fileProvider) ? ?{_fileProvider = fileProvider;} ? ?public IActionResult Index() ? ?{ ? ? ? ?var contents = _fileProvider.GetDirectoryContents(""); ? ? ? ?return View(contents);} }在應用的Startup類中創(chuàng)建provider的代碼如下:
using System.Linq;using System.Reflection;
using Microsoft.AspNetCore.Builder;
using Microsoft.AspNetCore.Hosting;
using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.FileProviders;
using Microsoft.Extensions.Logging;
namespace FileProviderSample{ ?
? ?public class Startup{ ? ? ?
? ? ?private IHostingEnvironment _hostingEnvironment; ? ? ? ?
? ?
? ? ?public Startup(IHostingEnvironment env) ? ? ? ?{ ?
? ? ?? ? ? ? ?var builder = new ConfigurationBuilder().SetBasePath(env.ContentRootPath).AddJsonFile("appsettings.json", optional: true, reloadOnChange: true).AddJsonFile($"appsettings.{env.EnvironmentName}.json", optional: true).AddEnvironmentVariables();Configuration = builder.Build();_hostingEnvironment = env;} ? ? ?
? ? ?? ? ? ? ?
? ? ? ?public IConfigurationRoot Configuration { get; } ? ? ? ?// This method gets called by the runtime. Use this method to add services to the container.
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?
public void ConfigureServices(IServiceCollection services) ? ? ? ?{ ? ? ? ? ? ?// Add framework services. ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? services.AddMvc(); ? ? ? ? ? ?var physicalProvider = _hostingEnvironment.ContentRootFileProvider; ? ? ? ? ? ?var embeddedProvider = new EmbeddedFileProvider(Assembly.GetEntryAssembly()); ? ? ? ? ? ?var compositeProvider = new CompositeFileProvider(physicalProvider, embeddedProvider); ? ? ? ? ? ?// choose one provider to use for the app and register it//services.AddSingleton<IFileProvider>(physicalProvider);//services.AddSingleton<IFileProvider>(embeddedProvider);services.AddSingleton<IFileProvider>(compositeProvider);}} } ? ? ?
在?Index.chhtml?視圖中,可以遍歷操作IDirectoryContents模型參數(shù)
@using Microsoft.Extensions.FileProviders @model ?IDirectoryContents<h2>Folder Contents</h2><ul> ? ?@foreach (IFileInfo item in Model) ? ?{ ? ? ? ?if (item.IsDirectory) ? ? ? ?{ ? ? ? ? ? ?<li><strong>@item.Name</strong></li> ? ? ? ?} ? ? ? ?else ? ? ? ?{ ? ? ? ? ? ?<li>@item.Name - @item.Length bytes</li> ? ? ? ?} ? ?}</ul>結果如下:
frameborder="0" scrolling="no" style="border-width: initial; border-style: none; width: 658px; height: 534px;">
EmbeddedFileProvider
EmbeddedFileProvider用于訪問嵌入到程序集中的文件。在.NET Core中,你可以通過修改?project.json?文件的buildOptions屬性參數(shù)來把文件嵌入到程序集中。
"buildOptions": { ?"emitEntryPoint": true, ?"preserveCompilationContext": true, ?"embed": [ ? ?"Resource.txt", ? ?"**/*.js"] }當你把文件嵌入到程序集中時,你可以使用通配符模式。這些模式可以被用來匹配一個或多個文件。
Note
把項目中所有的.js文件都嵌入到項目程序集里的情況是不太可能發(fā)生的,以上示例僅作為demo給出。
當創(chuàng)建一個EmbeddedFileProvider時,請在其構造函數(shù)中傳入一個程序集實例供其讀取。
var embeddedProvider = new EmbeddedFileProvider(Assembly.GetEntryAssembly());以上的代碼片段描述了如何創(chuàng)建一個能訪問當前工作程序集的EmbeddedFileProvider類型變量。
使用EmbeddedFileProvider更新示例項目代碼后的輸出結果如下:
frameborder="0" scrolling="no" style="border-width: initial; border-style: none; width: 658px; height: 534px;">
Note
如上圖所示,嵌入式資源不會公開目錄。相反的,資源路徑(經(jīng)由資源的命名空間)會被嵌入到它的文件名中并以.作為分隔符。
Tip
EmbeddedFileProvider構造器接受一個可選的baseNamespace參數(shù),指定此參數(shù)將限定GetDirectoryContents方法調用該命名空間下的資源。
CompositeFileProvider
CompositeFileProvider聯(lián)合IFileProvider實例公開了一個單一的接口,用以和來自多種provider的文件工作。當創(chuàng)建一個CompositeFileProvider時,你可以為它的構造函數(shù)傳入一個或多個IFileProvider實例。
var physicalProvider = _hostingEnvironment.ContentRootFileProvider;var embeddedProvider = new EmbeddedFileProvider(Assembly.GetEntryAssembly());var compositeProvider = new CompositeFileProvider(physicalProvider, embeddedProvider);使用包含物理式provider(在前)和嵌入式provider的CompositeFileProvider更新示例項目代碼后的輸出結果如下:
frameborder="0" scrolling="no" style="border-width: initial; border-style: none; width: 658px; height: 846px;">
查看更改
IFileProvider的Watch方法能用來查看一個或多個文件/目錄的更改信息。Watch方法接受一個路徑字符串,它也可以使用通配符模式來指定多個文件,Watch方法最終返回一個IChangeToken。這個token公開了一個HasChanged屬性用以檢視狀態(tài),公開了一個RegisterChangeCallback方法,此方法會在指定的路徑字符串檢測到更改時被調用。請注意每個更改token只調用其關聯(lián)回調以響應單次更改。為了使監(jiān)控持續(xù),你可以使用如下所示的TaskCompletionSource方法,或者重建IChangeToken以響應更改。
在這個文章的示例中,無論何時當文本文件內容發(fā)生修改,按如下代碼配置的console應用都會顯示相應的信息。
using System;using System.IO;
using System.Threading.Tasks;
using Microsoft.Extensions.FileProviders;
using Microsoft.Extensions.Primitives;
namespace WatchConsole{ ?
? ? public class Program{ ? ? ?
? ? ??private static PhysicalFileProvider _fileProvider = new PhysicalFileProvider(Directory.GetCurrentDirectory()); ? ? ? ?public static void Main(string[] args) ? ? ? ?{Console.WriteLine("Monitoring quotes.txt for changes (ctrl-c to quit)..."); ? ? ? ? ? ?while (true){MainAsync().GetAwaiter().GetResult();}} ? ? ?
? ? ??
? ? ???private static async Task MainAsync() ? ? ? ?{IChangeToken token = _fileProvider.Watch("quotes.txt"); ? ? ? ? ? ?var tcs = new TaskCompletionSource<object>();token.RegisterChangeCallback(state => ((TaskCompletionSource<object>)state).TrySetResult(null), tcs); ? ? ? ? ? ?await tcs.Task.ConfigureAwait(false);Console.WriteLine("quotes.txt changed");}} }
以下是執(zhí)行過幾次文本保存動作后的運行結果截圖:
frameborder="0" scrolling="no" style="border-width: initial; border-style: none; width: 658px; height: 345px;">
Note
有一些文件系統(tǒng),例如Docker容器和網(wǎng)絡共享,可能不能很可靠地發(fā)送更改通知。設置環(huán)境變量DOTNET_USE_POLLINGFILEWATCHER的值為1或true,使得每四秒輪詢一次文件系統(tǒng)的變更。
通配符模式
文件系統(tǒng)路徑規(guī)則使用叫作globbing patterns的通配符模式,這類簡單模式可以被用來指定文件組。這兩個通配符分別是*和**。
*
*表示在當前文件夾級別上匹配任何文件名稱或文件擴展名。匹配以文件路徑字符串中的/和.符號結尾。
**
**表示在多個目錄級別上匹配任何文件名稱或文件擴展名。可用于在一個目錄層次結構中遞歸地匹配多個文件。
通配符模式示例
directory/file.txt
在指定的文件夾中匹配指定的文件。
directory/*.txt
在指定的文件夾中匹配所有以.txt擴展名結尾的文件。
directory/*/project.json
在指定的directory文件夾下的一級目錄位置中匹配所有符合project.json名稱的文件
directory/**/*.txt
在指定的directory文件夾下的所有位置中匹配所有以.txt擴展名結尾的文件。
在ASP.NET Core中File Provider的用法
ASP.NET Core有幾個組件使用file provider功能。IHostingEnvironment以IFileProvider接口類型公開了應用的目錄根和Web根。靜態(tài)文件中間件使用file provider來定位靜態(tài)文件。Razor更是大量使用IFileProvider來定位視圖。Dotnet的發(fā)布功能使用file provider和通配符模式來指定需要跟隨發(fā)布的文件。
在應用程序中使用的建議
如果你的ASP.NET Core應用需要訪問文件系統(tǒng),你可以通過依賴注入創(chuàng)建IFileProvider接口實例,然后再通過前文所示的相應方法執(zhí)行訪問。當應用啟動的時候,這些方法允許你一次性配置provider并減少應用初始化時生成的實例類型數(shù)目。
原文地址:http://www.cnblogs.com/Wddpct/p/6128386.html
.NET社區(qū)新聞,深度好文,微信中搜索dotNET跨平臺或掃描二維碼關注
總結
以上是生活随笔為你收集整理的ASP.NET Core File Providers的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 微服务的前世今生
- 下一篇: 关于全局ID,雪花(snowflake)