前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >专栏 >关于 .NET 在不同操作系统中 IO 文件路径拼接方法,升级 .NET 7 后注意到的一个知识点

关于 .NET 在不同操作系统中 IO 文件路径拼接方法,升级 .NET 7 后注意到的一个知识点

作者头像
乌拉栋
发布于 2022-11-30 07:24:20
发布于 2022-11-30 07:24:20
1.3K0
举报
文章被收录于专栏:编程宝典编程宝典

.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

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2022-11-24,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
C# 基础知识系列- 14 IO篇 文件的操作 (3)
Path的中文名称有路径的意思,所以Path类就是路径类,C#把Path设置为工具类,路径的实例被区分为文件和目录了。以下是它的定义:
程序员小高
2020/05/14
6780
C# 基础知识系列- 14 IO篇 文件的操作 (3)
.NET 将混合了多个不同平台(Windows / Mac / Linux)的文件/目录的路径格式化成同一个平台下的路径
Windows 下的路径分隔符是 \ 而 Linux 和 Mac 下的路径分隔符是 \。正常如果你的数据不跨 Windows 和 Linux 平台流通的话,不怎么会遇到多种换行符并存的问题的。但如果真发生了流通,那么如何将它们格式化为统一的当前平台认识的分隔符呢?
walterlv
2023/10/22
4410
.NET 将混合了多个不同平台(Windows / Mac / Linux)的文件/目录的路径格式化成同一个平台下的路径
浅谈.Net Framework中实现的文件操作(二)
File类是一个工具类,它可以便捷的通过文件路径,对文件进行一系列的操作。上一篇文章里面介绍了FileInfo类的用法,File类里面的很多功能和FileInfo提供的功能类似,并提供了更多的方法,通过查看.Net Framework源码,FileInfo类里面有调用File的方法,但是File里面引用到FileInfo类。
小蜜蜂
2019/07/17
3690
浅谈.Net Framework中实现的文件操作(二)
【愚公系列】2023年01月 .NET/C#知识点-文件上传与下载
文章目录 前言 一、文件上传与下载 1.单文件上传 2.多文件上传 3.文件下载 相关类库 ---- 前言 文件上传有两种方式:WEB上传和FTP上传。 WEB上传就是通过点击网页中的“浏览”、“选定”、“上传”(或“确定”、“提交”)等按钮来上传文件的方式。精炼地说就是通过网页的功能上传文件。 FTP[File Transfer Protocol]中文译为文件传输协议,是Internet上的另一项主要服务,这项服务让使用者能通过Internet来传输各式各样的文件。FTP上传的目标服务器有一个固定的FT
愚公搬代码
2023/03/16
4520
【愚公系列】2023年01月 .NET/C#知识点-文件上传与下载
ASP.NET Core WEB API 使用element-ui文件上传组件el-upload执行手动文件文件,并在文件上传后清空文件
  从开始学习Vue到使用element-ui-admin已经有将近快两年的时间了,在之前的开发中使用element-ui上传组件el-upload都是直接使用文件选取后立即选择上传,今天刚好做了一个和之前类似的文件选择上传的需求,不过这次是需要手动点击按钮把文件上传到服务器中进行数据导入,而且最多只能够选择一个文件进行上传,上传成功后需要对file-list中的文件列表数据进行清空操作,在这里服务端使用的是ASP.NET Core WEB API来进行文件流数据接收和保存。
追逐时光者
2020/04/16
2.5K0
用pathlib进行Python中的文件路径处理
本文介绍Python中的文件路径处理方法,从字符串连接、os.path.join()到Python3中处理文件路径的简单方法:pathlib。
MeteoAI
2019/07/24
4K0
os.path:Python操作和处理文件路径
os.path是平台独立的文件名管理库,使用该库能够很方便来处理多个平台上的文件。即使程序不打算在平台之间移值,也应当使用os.path库来完成可靠的文件名解析。
luckpunk
2023/09/27
4700
os.path:Python操作和处理文件路径
课时30:文件系统:介绍一个高大上的东西
接下来会介绍跟Python的文件相关的一些很有用的模块。模块是什么?其实我们写的每一个源代码文件(*.py)都是一个模块。Python自身带有非常多使用的模块。
py3study
2020/01/19
5570
面向对象(二十)-文件操作
项目中的文件操作,要求将项目中添加的文件放到项目输出目录下。 首先将文件拖拽到项目目录下。然后如下操作:
孙寅
2020/06/02
4350
【文件IO】文件系统操作
Java 中,提供了一个 File 类,进行文件系统操作,这个对象会使用“路径”进行初始化,从而表示一个具体的文件(这个文件可以存在,也可以不存在) 再基于这个对象进行后续操作
椰椰椰耶
2024/10/15
1460
【文件IO】文件系统操作
.NET开源工具类库-Masuit.Tools
(https://www.nuget.org/packages/Masuit.Tools.Core) 包含一些常用的操作类,大都是静态类,加密解密,反射操作,动态编译,权重随机筛选算法,简繁转换,分布式短id,表达式树,linq扩展,文件压缩,多线程下载和FTP客户端,硬件信息,字符串扩展方法,日期时间扩展操作,中国农历,大文件拷贝,图像裁剪,验证码,断点续传,实体映射、集合扩展等常用封装。 官网教程
ccf19881030
2020/10/26
2.6K0
.NET开源工具类库-Masuit.Tools
疯狂吐槽 MAUI 以及 MAUI 入坑知识点
目前 .NET 已经升级到 7.0,但是 MAUI 还是一如既往的拉跨,如果开发过 MAUI,做过定制,自定义标题栏之类的,便会发现 MAUI 有多难受。
痴者工良
2023/02/02
6K4
疯狂吐槽 MAUI 以及 MAUI 入坑知识点
【javaEE】文件操作--io
E:\cs2\5EClient 中的 \ 就是 pathSeparator,如果当前的系统是 Windows,\ 或者 / 都可以作为分隔符,如果系统是 Linux 或 Mac ,只能使用 / 作为分隔符,一般我们都建议使用 / 作为分隔符,因为 \ 一般还需要搭配转义字符来使用,而且因为linux也是/,如果在windows中用/可以直接跨系统。
E绵绵
2025/03/14
570
【javaEE】文件操作--io
python 文件和路径操作函数小结
1: os.listdir(path) //path为目录  功能相当于在path目录下执行dir命令,返回为list类型  print os.listdir('..')  2: os.path.walk(path,visit,arg)  path :是将要遍历的目录  visit :是一个函数指针,函数圆形为:  callback(arg,dir,fileList)  其中arg为为传给walk的arg , dir是path下的一个目录,fileList为dir下的文件和目录组成的list, arg:传给visit用的  3:os.path.split(path)  path 为一个路径,  输出,把path分成两部分,具体看实例:  print os.path.split("abc/de.txt")  ('abc', 'de.txt')  os.path.split("abc")  (", 'abc')  print os.path.split("de/abc/de")  ('de/abc', 'de')  4: os.path.splitext(filename)  把文件名分成文件名称和扩展名  os.path.splitext(abc/abcd.txt)  ('abc/abcd', '.txt')  5: os.path.dirname(path)  把目录名提出来  print os.path.dirname("abc")  #输出为空  print os.path.dirname('abc\def')  abc  6: os.path.basename(filename)  取得主文件名  print os.path.basename('abc')  abc  print os.path.basename('abc.txt')  abc  print os.path.basename('bcd/abc')  abc #这个需要注意不包括目录名称  print os.path.basename('.')  .  7:os.mkdir(path, [mode])  path为目录名: 这里有个要求,只能创建一级目录  比如path为 abc/def 则当前目录下必须存在abc 否则失败  8: os.makedirs(path [,mode])  可以创建多级目录  9:os.remove(path)删除一个文件,一定是一个文件  os.removedirs(path) 删除一个目录下所有东西  os.rmdir(path) 删除一个目录,而且一定要空,否则os.errer  10:os.walk(path)  遍历path,返回一个对象,他的每个部分都是一个三元组  ('目录x',[目录x下的目录list],目录x下面的文件)  举例:  a = os.walk('.')  for i in a:  print i  11:shutil.copy(src,dst)  把文件src内容拷贝到文件dst中。,目标区域必须可以写,如果dst存在,则dst被覆盖  ――――――――――――――――――  11.1 os.path -- 普通的路径名操作  这个模块实现一些在路径名称上有用的函数。  警告:在 Windows 上,这些函数中的多个不能正确地支持 UNC 路径名。splitunc() 和 ismount() 可以正确地处理它们。  abspath(path)  返回 path 路径名一个规格化的绝对化的版本。在大多数平台上,相当于normpath(join(os.getcwd(), path))。1.5.2版本中的新特性。  basename(path)  返回 path 路径名最底部的名称。这是由 split(path)返回的对的第二半部分。注意这个函数的结果不同于Unix basename 程序;UNIX的 basename对于'/foo/bar/' 返回 'bar',basename() 函数返回一个空字符串('')。  commonprefix(list)  返回最长的路径前缀(采取字符的字符),是 list中所有路径的一个前缀。如果 list 为空,返回空字符串('')。注意可以返回无效的路径因为它每次操作一个字符。  dirname(path)  返回 path 路径名的目录名称。这是由 split(path)返回的对的第一半部分。  exists(path)  如果 path 与一个已存在的路径关联返回True。对于损坏的符号连接返回 False 。在一些平台上,如果权限没有被授予在请求的文件上执行os.stat()这个函数可能返回False,即使该 path 实际上存在。  lexists(path)  如果 path 与一个已存在的路径关联返回 True 。对于损坏的连接返回 True
py3study
2020/01/09
1.8K0
【愚公系列】2021年12月 Python教学课程 26-系统编程
如果我们要操作文件、目录,可以在命令行下面输入操作系统提供的各种命令来完成。比如 dir,cd 等命令。如果要在 Python 程序中执行这些目录和文件的操作怎么办?其实操作系统提供的命令只是简单地调用了操作系统提供的接口函数,Python 内置的 os模块也可以直接调用操作系统提供的接口函数。
愚公搬代码
2021/12/16
8870
【愚公系列】2021年12月 Python教学课程 26-系统编程
python之调用系统命令 原
os模块包装了不同操作系统的通用接口,使用户在不同操作系统下,可以使用相同的函数接口,返回相同结构的结果。
晓歌
2018/08/15
2.3K0
python之调用系统命令
                                                                            原
【JavaEE初阶】文件操作——IO
文件:File这个概念,在计算机里也是一次多用: 狭义的文件:指的是硬盘上的文件和目录。针对硬盘这种持久化存储的I/O设备,当我们想要进行数据保存时,往往不是保存成一个整体,而是独立成一个个的单位进行保存,这个独立的单位就被抽象成文件的概念,就类似办公桌上的一份份真实的文件一般。 广义的文件:泛指计算机中的很多的软硬件资源.
xxxflower
2023/10/16
1470
【JavaEE初阶】文件操作——IO
路径,文件,目录,I/O常见操作汇总
摘要:   文件操作是程序中非常基础和重要的内容,而路径、文件、目录以及I/O都是在进行文件操作时运维
Java架构师必看
2021/03/22
1.5K0
附实战代码|告别OS模块,体验Python文件操作新姿势!
相比常用的 os.path而言,pathlib 对于目录路径的操作更简介也更贴近 Pythonic。但是它不单纯是为了简化操作,还有更大的用途。
刘早起
2020/11/23
5660
附实战代码|告别OS模块,体验Python文件操作新姿势!
带外攻击OOB(RCE无回显骚思路总结)
向服务器提交一个 payload,而服务器响应给我们相关的 response 信息。大家都叫它带内攻击,这些理论的东西,我们简单理解就好,这里我们就理解成单挑通信的通道为带内攻击,也就是整个测试过程或者说是交互过程,中间没有其外部的服务器参与,只有自己和目标服务器,那么就叫带内。
枪哥四海为家
2022/03/14
5.8K0
带外攻击OOB(RCE无回显骚思路总结)
推荐阅读
相关推荐
C# 基础知识系列- 14 IO篇 文件的操作 (3)
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
查看详情【社区公告】 技术创作特训营有奖征文