關於 .NET 在不同操作系統中 IO 文件路徑拼接方法,升級 .NET 7 後註意到的一個知識點

来源:https://www.cnblogs.com/berkerdong/archive/2022/11/24/16920239.html
-Advertisement-
Play Games

.NET 現在支持跨平臺這件事情已經是眾所周知的特點了,雖然平臺整體支持跨平臺了,但是我們的代碼如果真的想要實現跨平臺運行其實還是有些小細節要註意的,今天想要記錄分享的就是關於 文件I/O操作時路徑的拼接問題。 在 Windows 環境下我們常見的路徑格式如下: D:\Software\AppDat ...


.NET 現在支持跨平臺這件事情已經是眾所周知的特點了,雖然平臺整體支持跨平臺了,但是我們的代碼如果真的想要實現跨平臺運行其實還是有些小細節要註意的,今天想要記錄分享的就是關於 文件I/O操作時路徑的拼接問題。

在 Windows 環境下我們常見的路徑格式如下:

D:\Software\AppData\Files\aaa.jpg

可以看到 Windows 環境下文分隔符為 \ 路徑由三部分組成分別是:

  1. 盤符: D:\
  2. 文件夾層級:Software\AppData\Files
  3. 文件名:aaa.jpg

在 .NET 平臺常見的獲取當成程式主機路徑的方法主要從

.NET 控制台程式,通過依賴註入獲取 IHostEnvironment hostEnvironment

.NET Web程式,通過依賴註入獲取 IWebHostEnvironment webHostEnvironment

hostEnvironment.ContentRootPath
webHostEnvironment.ContentRootPath

ContentRootPath 指的是應用程式內容文件的目錄的絕對路徑;


webHostEnvironment.WebRootPath

WebRootPath 指的是 其實就是用於存放靜態資源的那個 wwwroot 目錄的絕對路徑,ASP.NET Core MVC 項目的 css、 js、 img 等靜態資源一般都是存放在 wwwroot 目錄中,ASP.NET Core WebAPI 項目有需要也可以開啟這個 wwwroot 的選項,只要在項目啟動的時候 app.UseStaticFiles(); 啟用靜態文件模塊即可。


在剛開始接觸 .NET 項目時,我代碼中的文件上傳路徑是這樣拼接的。

webHostEnvironment.ContentRootPath + "files\\"+ DateTime.UtcNow.ToString("yyyy\\MM\\dd\\")+"xxx.jpg";

這樣組合出來的路徑地址可能如下:

d:\appdata\files\2022\11\24\xxx.jpg

如果代碼這樣寫,我們在 Windows 平臺運行是不會有有任何問題的,但是如果有一天想要嘗試跨平臺部署,把代碼搬到 Linux 或者 Mac 平臺運行就會發現這個代碼會報錯,原因在於 Linux 和 Mac 平臺無法識別 \ 分割憑藉的文件路徑,因為這兩個平臺是採用 / 做為文件路徑分割符的。

比如 Linux 下的常見路徑格式如下:

/var/appdata/xxxx

所以這個時候我們只要調整我們的代碼為

webHostEnvironment.ContentRootPath + "files/"+ DateTime.UtcNow.ToString("yyyy/MM/dd/")+"xxx.jpg";

拼接出來的路徑格式則為

d:/appdata/files/2022/11/24/xxx.jpg

/var/appdata/files/2022/11/24/xxx.jpg

重新編譯之後就可以在 Linux 和 Mac 平臺運行了,並且 Windows 平臺其實也是可以相容 / 作為文件路徑分割符號的,至此三個平臺都可以正常運行了。

上面的代碼運行了3年左右時間,直至最近更新了 .NET 7 發現上面的代碼,在伺服器上又報錯了,上面的代碼執行效果變成了下麵這樣

d:/appdatafiles/2022/11/24/xxx.jpg

/var/appdatafiles/2022/11/24/xxx.jpg

通過觀察可以發現原來是 appdata/files 之間的 分隔符 / 消失了,導致拼接的結果變成了 appdatafiles ,經過調試之後發現原因如下:
在 .NET 6.0 及以前的版本中

webHostEnvironment.ContentRootPath;
webHostEnvironment.WebRootPath;
hostEnvironment.ContentRootPath;

三個變數的末尾都是帶有一個分隔符的,他們的取值都是

d:/appdata/var/appdata/ 像這樣尾部有跟隨一個 / 分割符,但是到了 .NET 7.0 中,他們的取值變了,變成了
d:/appdatavar/appdata 尾部的分割符號不見了,這就導致我們上面的路徑拼接代碼出現了異常。

這時候想起來微軟官方自帶的拼接方法 Path.Combine ,該方法用於將多個路徑信息進行拼接,改造後的代碼如下

Path.Combine(webHostEnvironment.ContentRootPath, "files", DateTime.UtcNow.ToString("yyyy"),DateTime.UtcNow.ToString("MM"),DateTime.UtcNow.ToString("dd"),"xxx.jpg");
這樣的到結果如下

d:\appdata\files\2022\11\24\xxx.jpg

/var/appdata/files/2022/11/24/xxx.jpg

可以看到在 Windows 平臺運行時還是採用了預設的 \ 作為文件夾的分割符號,而在 Linux 和 Mac 平臺運行時則採用了 / 作為文件夾的分割符號。

雖然通過 Path.Combine 可以自動生成符合各個平臺運行要求的路徑,倒是如果需要把文件路徑保存起來的時候還是建議採用 / 作為文件分隔符,這樣方便隨時切換運行平臺,否則 代碼在 Windows 平臺運行期間產生的數據保存到資料庫之後,將來有一天切換到其他平臺時這樣的路徑被查詢出來執行時還是會報錯,但是採用 / 作為文件分隔符則不需要擔心,所以像文件上傳方法這種場景在需要記錄文件路徑到資料庫時可以 .Replace("\","/") 對路徑進行一下轉換之後再保存到資料庫中

Path.Combine(webHostEnvironment.ContentRootPath, "files", DateTime.UtcNow.ToString("yyyy"),DateTime.UtcNow.ToString("MM"),DateTime.UtcNow.ToString("dd"),"xxx.jpg").Replace("\\","/");


可能有人會問,為什麼 Windows 就不能和 Mac 與 Linux 等系統一樣本身也預設採用 / 作為文件分隔符,直接大統一多好,其實這屬於歷史遺留問題了,因為在 Windows 平臺還是 DOS 的時候,那個時候 / 在 Windows 平臺是作為命令的參數標記使用的,所以為了不和 命令參數符號 / 重覆,就採用最為接近的 \ 充當了路徑分隔符,而 Linux 與 Mac 平臺傳遞參數則是採用 - 符號,如我們熟知的 ipconfig 命令。

預設查詢的簡單信息,如果需要查詢全部信息則是
ipconfig /all

如果需要清理 dns 緩存信息則是
ipconfig /flushdns

可以看到傳遞參數時是需要 / 符號的,當然現在新版的 Windows 系統其實也支持 - 作為參數傳遞符號了,下麵的命令也可以正常運行
ipconfig -all
ipconfig -flushdns

至此 關於 .NET 在不同操作系統中 IO 文件路徑拼接方法總結 就講解完了,有任何不明白的,可以在文章下麵評論或者私信我,歡迎大家積極的討論交流,有興趣的朋友可以關註我目前在維護的一個 .NET 基礎框架項目,項目地址如下
https://github.com/berkerdong/NetEngine.git
https://gitee.com/berkerdong/NetEngine.git


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

-Advertisement-
Play Games
更多相關文章
  • 接上篇: 通過位元組碼,我們瞭解了class文件的結構 通過運行數據區,我們瞭解了jvm內部的記憶體劃分及結構 接下來,讓我們看看,位元組碼怎麼進入jvm的記憶體空間,各自進入那個空間,以及怎麼跑起來。 4.1 載入 4.1.1 概述 類的載入就是將class文件中的二進位數據讀取到記憶體中,然後將該位元組流所 ...
  • 主要推到了極化碼編碼矩陣生成迭代方式,並針對遞歸方法和按位生成(硬體生成不適用遞歸方案)的方法用matlab實現。 通道組合 W表示原始B-DMC通道。 下圖是兩個通道組合的例子。 長度為2的通道組合模型 長度為4的通道組合模型 長度為N/2與N的通道組合形式 G的推導及性質 G公式推導 編碼矩陣生 ...
  • 對於緩存容器而言,容量限制與數據淘汰是兩個基礎且核心的關鍵點,也是實際使用的時候使用頻率最高的特性。本篇在上一文基礎上深入解讀下Guava Cache中的容量限制與數據淘汰策略的實現與使用約束。 ...
  • Spring,作為 Java EE 的事實規範,在2022年11月16日發佈了最新的 6.0.0 GA 版本。這個版本是框架後續新生代的初始版本,擁抱持續創新的 OpenJDK 和 Java 生態。新的版本以 Java 17+ 作為 baseline,並遷移至 Jakarta EE 9+(即,使用 ...
  • 作者:寧海翔 1 前言 對象拷貝,是我們在開發過程中,繞不開的過程,既存在於Po、Dto、Do、Vo各個表現層數據的轉換,也存在於系統交互如序列化、反序列化。 Java對象拷貝分為深拷貝和淺拷貝,目前常用的屬性拷貝工具,包括Apache的BeanUtils、Spring的BeanUtils、Cgli ...
  • 用python爬蟲技術,爬取百度搜索結果數據,包含欄位: 頁碼、標題、百度鏈接、真實鏈接、簡介、網站名稱。 並把源碼封裝成exe文件,方便沒有python環境,或者不懂技術的人使用它。 ...
  • 上篇隨筆發佈後,一天內,十幾個爬蟲網站爬取了我的隨筆,有些網站非但沒有註明來源,反而將自己標為博文的原創者,並更改了圖像水印,這篇被爬的隨筆雖瀏覽量不大,但好歹是自己一字一字認真碼出來的,被偷實在令人心情不爽,在本篇隨筆記錄一下。 ...
  • WebDAV 是超文本傳輸協議 (HTTP) 的一組擴展,為 Internet 上電腦之間的編輯和文件管理提供了標準.利用這個協議用戶可以通過Web進行遠程的基本文件操作,如拷貝、移動、刪除等。 在IIS 7.0中,WebDAV是作為獨立擴展模塊,需要單獨進行下載,而IIS 7.5以及以上版本中... ...
一周排行
    -Advertisement-
    Play Games
  • Timer是什麼 Timer 是一種用於創建定期粒度行為的機制。 與標準的 .NET System.Threading.Timer 類相似,Orleans 的 Timer 允許在一段時間後執行特定的操作,或者在特定的時間間隔內重覆執行操作。 它在分散式系統中具有重要作用,特別是在處理需要周期性執行的 ...
  • 前言 相信很多做WPF開發的小伙伴都遇到過表格類的需求,雖然現有的Grid控制項也能實現,但是使用起來的體驗感並不好,比如要實現一個Excel中的表格效果,估計你能想到的第一個方法就是套Border控制項,用這種方法你需要控制每個Border的邊框,並且在一堆Bordr中找到Grid.Row,Grid. ...
  • .NET C#程式啟動閃退,目錄導致的問題 這是第2次踩這個坑了,很小的編程細節,容易忽略,所以寫個博客,分享給大家。 1.第一次坑:是windows 系統把程式運行成服務,找不到配置文件,原因是以服務運行它的工作目錄是在C:\Windows\System32 2.本次坑:WPF桌面程式通過註冊表設 ...
  • 在分散式系統中,數據的持久化是至關重要的一環。 Orleans 7 引入了強大的持久化功能,使得在分散式環境下管理數據變得更加輕鬆和可靠。 本文將介紹什麼是 Orleans 7 的持久化,如何設置它以及相應的代碼示例。 什麼是 Orleans 7 的持久化? Orleans 7 的持久化是指將 Or ...
  • 前言 .NET Feature Management 是一個用於管理應用程式功能的庫,它可以幫助開發人員在應用程式中輕鬆地添加、移除和管理功能。使用 Feature Management,開發人員可以根據不同用戶、環境或其他條件來動態地控制應用程式中的功能。這使得開發人員可以更靈活地管理應用程式的功 ...
  • 在 WPF 應用程式中,拖放操作是實現用戶交互的重要組成部分。通過拖放操作,用戶可以輕鬆地將數據從一個位置移動到另一個位置,或者將控制項從一個容器移動到另一個容器。然而,WPF 中預設的拖放操作可能並不是那麼好用。為瞭解決這個問題,我們可以自定義一個 Panel 來實現更簡單的拖拽操作。 自定義 Pa ...
  • 在實際使用中,由於涉及到不同編程語言之間互相調用,導致C++ 中的OpenCV與C#中的OpenCvSharp 圖像數據在不同編程語言之間難以有效傳遞。在本文中我們將結合OpenCvSharp源碼實現原理,探究兩種數據之間的通信方式。 ...
  • 一、前言 這是一篇搭建許可權管理系統的系列文章。 隨著網路的發展,信息安全對應任何企業來說都越發的重要,而本系列文章將和大家一起一步一步搭建一個全新的許可權管理系統。 說明:由於搭建一個全新的項目過於繁瑣,所有作者將挑選核心代碼和核心思路進行分享。 二、技術選擇 三、開始設計 1、自主搭建vue前端和. ...
  • Csharper中的表達式樹 這節課來瞭解一下表示式樹是什麼? 在C#中,表達式樹是一種數據結構,它可以表示一些代碼塊,如Lambda表達式或查詢表達式。表達式樹使你能夠查看和操作數據,就像你可以查看和操作代碼一樣。它們通常用於創建動態查詢和解析表達式。 一、認識表達式樹 為什麼要這樣說?它和委托有 ...
  • 在使用Django等框架來操作MySQL時,實際上底層還是通過Python來操作的,首先需要安裝一個驅動程式,在Python3中,驅動程式有多種選擇,比如有pymysql以及mysqlclient等。使用pip命令安裝mysqlclient失敗應如何解決? 安裝的python版本說明 機器同時安裝了 ...