ASP.NET Core File Providers

来源:http://www.cnblogs.com/Wddpct/archive/2016/12/03/6128386.html
-Advertisement-
Play Games

"原文地址:FileProvider" By [Steve Smith][1] ASP.NET Core通過對File Providers的使用實現了對文件系統訪問的抽象。 [查看或下載示例代碼][2] File Provider 抽象 File Providers是文件系統之上的一層抽象。它的主要 ...


原文地址:FileProvider

By Steve Smith

ASP.NET Core通過對File Providers的使用實現了對文件系統訪問的抽象。

查看或下載示例代碼

File Provider 抽象

File Providers是文件系統之上的一層抽象。它的主要介面是IFileProviderIFileProvider公開了相應方法用來獲取文件信息(IFileInfo), 目錄信息(IDirectoryContents),以及設置更改通知(通過使用一個IChangeToken)。

IFileInfo介面提供了操作單個文件和目錄的方法和屬性。它有兩個boolean屬性,ExistsIsDirectory,以及兩個描述文件的兩個屬性NameLength(按位元組),還包括一個LastModified日期屬性。你還可以通過CreateReadStream方法讀取文件內容。

File Provider 實現

有三種對於IFileProvider的實現可供選擇:物理式,嵌入式和複合式。物理式用於訪問實際系統中的文件。嵌入式用於訪問嵌入在程式集中的文件。 複合式則是對前兩種方式的組合使用。

PhysicalFileProvider

PhysicalFileProvider提供了對物理文件系統的訪問。它封裝了System.IO.File類型,範圍限定到一個目錄及其子目錄的所有路徑。這類作用域會限制訪問某個目錄及其子目錄,防止作用域以外的其他操作訪問文件系統。當實例化此類provider時,你必須為它提供一個目錄路徑,以供伺服器拿來當做由這個provider發出的所有請求的基礎路徑(這個provider會限制路徑以外的訪問請求)。在一個ASP.NET Core應用,你可以直接實例化出一個PhysicalFileProvider provider,或者你也可以通過在控制器和服務中使用構造函數依賴註入的方式,請求一個IFileProvider介面。後者生成的解決方案通常更靈活以及更便於測試。

要創建一個PhysicalFileProvider其實很簡單,只需要對其實化,再傳遞給它一個物理路徑。之後你就可以通過它的目錄遍歷內容或提供子路徑獲取特定文件的信息。

IFileProvider provider = new PhysicalFileProvider(applicationRoot);
IDirectoryContents contents = provider.GetDirectoryContents(""); // the applicationRoot contents
IFileInfo fileInfo = provider.GetFileInfo("wwwroot/js/site.js"); // a file under applicationRoot

為了在控制器中請求一個provider,需要在控制器的構造函數中指定類型參數並賦值給本地屬性。之後你就可以在你的動作器方法中使用本地實例了。

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類中創建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模型參數

@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>

結果如下:

EmbeddedFileProvider

EmbeddedFileProvider用於訪問嵌入到程式集中的文件。在.NET Core中,你可以通過修改 project.json 文件的buildOptions屬性參數來把文件嵌入到程式集中。

"buildOptions": {
  "emitEntryPoint": true,
  "preserveCompilationContext": true,
  "embed": [
    "Resource.txt",
    "**/*.js"
  ]
}

當你把文件嵌入到程式集中時,你可以使用通配符模式。這些模式可以被用來匹配一個或多個文件。

Note
把項目中所有的.js文件都嵌入到項目程式集里的情況是不太可能發生的,以上示例僅作為demo給出。

當創建一個EmbeddedFileProvider時,請在其構造函數中傳入一個程式集實例供其讀取。

var embeddedProvider = new EmbeddedFileProvider(Assembly.GetEntryAssembly());

以上的代碼片段描述瞭如何創建一個能訪問當前工作程式集的EmbeddedFileProvider類型變數。

使用EmbeddedFileProvider更新示例項目代碼後的輸出結果如下:

Note
如上圖所示,嵌入式資源不會公開目錄。相反的,資源路徑(經由資源的命名空間)會被嵌入到它的文件名中並以.作為分隔符。

Tip
EmbeddedFileProvider構造器接受一個可選的baseNamespace參數,指定此參數將限定GetDirectoryContents方法調用該命名空間下的資源。

CompositeFileProvider

CompositeFileProvider聯合IFileProvider實例公開了一個單一的介面,用以和來自多種provider的文件工作。當創建一個CompositeFileProvider時,你可以為它的構造函數傳入一個或多個IFileProvider實例。

var physicalProvider = _hostingEnvironment.ContentRootFileProvider;
var embeddedProvider = new EmbeddedFileProvider(Assembly.GetEntryAssembly());
var compositeProvider = new CompositeFileProvider(physicalProvider, embeddedProvider);

使用包含物理式provider(在前)和嵌入式provider的CompositeFileProvider更新示例項目代碼後的輸出結果如下:

查看更改

IFileProviderWatch方法能用來查看一個或多個文件/目錄的更改信息。Watch方法接受一個路徑字元串,它也可以使用通配符模式來指定多個文件,Watch方法最終返回一個IChangeToken。這個token公開了一個HasChanged屬性用以檢視狀態,公開了一個RegisterChangeCallback方法,此方法會在指定的路徑字元串檢測到更改時被調用。請註意每個更改token只調用其關聯回調以響應單次更改。為了使監控持續,你可以使用如下所示的TaskCompletionSource方法,或者重建IChangeToken以響應更改。

在這個文章的示例中,無論何時當文本文件內容發生修改,按如下代碼配置的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");
        }
    }
}

以下是執行過幾次文本保存動作後的運行結果截圖:

Note
有一些文件系統,例如Docker容器和網路共用,可能不能很可靠地發送更改通知。設置環境變數DOTNET_USE_POLLINGFILEWATCHER的值為1true,使得每四秒輪詢一次文件系統的變更。

通配符模式

文件系統路徑規則使用叫作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功能。IHostingEnvironmentIFileProvider介面類型公開了應用的目錄根和Web根。靜態文件中間件使用file provider來定位靜態文件。Razor更是大量使用IFileProvider來定位視圖。Dotnet的發佈功能使用file provider和通配符模式來指定需要跟隨發佈的文件。

在應用程式中使用的建議

如果你的ASP.NET Core應用需要訪問文件系統,你可以通過依賴註入創建IFileProvider介面實例,然後再通過前文所示的相應方法執行訪問。當應用啟動的時候,這些方法允許你一次性配置provider並減少應用初始化時生成的實例類型數目。




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

-Advertisement-
Play Games
更多相關文章
  • 1.確認ip地址、子網掩碼、網關是正確的。 ifconfig 2.確認區域網是互通的,訪問別人的電腦、網關 ping 發送數據包接收數據包,設備是否聯通 /etc/sysconfig/network-scripts 修改網卡配置 3.確定解析功能變數名稱正確 nslookup 配置dns: vi /etc/ ...
  • 今天遇到一個需求,gird表格數據如下: 實際需要顯示的結果為: 上述需求一般有三種處理方式: 1.資料庫直接生成分組數據。該方式從數據源頭進行處理,好處是不需要在DW視窗做分組處理,不好還處是會在資料庫中重覆檢索生成合計、小計數據,不利於性能優化。 2.使用DW的group DW製作分組顯示界面, ...
  • 1.安裝ubuntu時使用的virt-install的配置: 報錯如下: 通過查資料發現,virt-install可以開debug模式的,加入--debug選項即可 2.virt-install的debug模式得到的結果: 這裡就可以看出問題了,明明是64位的操作系統,為什麼去找./install/ ...
  • 最近悟出來一個道理,在這兒分享給大家:學歷代表你的過去,能力代表你的現在,學習代表你的將來。 十年河東十年河西,莫欺少年窮 學無止境,精益求精 上篇博客我們學習了EF 之 MVC 排序,查詢,分頁 Sorting, Filtering, and Paging For MVC About EF,本節繼 ...
  • 從資料庫獲取構造樹結構是ExtJS TreePanel的核心技術,常用方法是TreeStroe里配置proxy,這種方式的root成了一個不受控制的節點。 TreeStroe的root實際是一個層疊json數據,大部分情況是直接寫一些簡單數據,但在實際應用中必定是要從資料庫讀取的。我的方法是先用Ex... ...
  • 代碼: using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.IO; using Syste ...
  • 應用場景 應用場景 項目開發決定使用angular2進行前後端分離開發,由我負責後端服務的開發,起初選擇的是web api進行開發。對跨域訪問通過API中間件+過濾器對跨域訪問進行支持。開發一段後,通知需要移植到MVC4項目中一同發佈angular2並且放棄API,但前期開發仍然需要分離開發。 遇到 ...
  • 今天在做發送郵件功能時,開始用qq郵箱和163郵箱都可以正常發送,後再改用我公司的郵箱和smtp時竟然報錯了。 異常提示 “根據驗證過程,遠程證書無效”,後來通過查詢資料解決該問題,上代碼: ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...