当前位置:首页 > 科技  > 软件

Go 中如何检查文件是否存在?可能产生竞态条件?

来源: 责编: 时间:2024-02-05 17:20:09 318观看
导读Go 中如何检查文件是否存在呢?如果你用的是 Python,可通过 os.path.exists 这样的标准库函数实现。遗憾的是,Go 标准库没有提供这样直接的函数,但好在,没有直接的,却有不那么直接的方法。本文将基于这个话题展开,介绍 Go 中

Go 中如何检查文件是否存在呢?8BX28资讯网——每日最新资讯28at.com

如果你用的是 Python,可通过 os.path.exists 这样的标准库函数实现。遗憾的是,Go 标准库没有提供这样直接的函数,但好在,没有直接的,却有不那么直接的方法。8BX28资讯网——每日最新资讯28at.com

本文将基于这个话题展开,介绍 Go 中如何检查文件是否存在。8BX28资讯网——每日最新资讯28at.com

另外,本文最后还会介绍一个小注意点,即在判断文件是否存在时,如何避免中潜在的竞态条件。8BX28资讯网——每日最新资讯28at.com

os.Stat 检查文件状态

Go 标准库虽然没有提供类似于 os.Exist 这样直接的函数检查文件是否存在,但它提供另外一个函数 os.Stat。8BX28资讯网——每日最新资讯28at.com

os.Stat 函数的作用是获取文件状态信息,我们通过检查它返回的错误即可知晓文件是否存在。8BX28资讯网——每日最新资讯28at.com

图片图片8BX28资讯网——每日最新资讯28at.com

示例代码,如下所示:8BX28资讯网——每日最新资讯28at.com

func main() {  _, err := os.Stat("/path/to/file")  if err != nil {    if os.IsNotExist(err) {      // 文件不存在    } else {      // 其他错误    }  }  // 文件存在}

第一个返回值表示文件信息,不是我们关心的重点,直接省略掉。8BX28资讯网——每日最新资讯28at.com

第二个返回值表示错误 error。如果文件不存在,可通过检查 os.IsNotExist 检查 error 是否是 os.ErrNotExist,确定文件是否存在。8BX28资讯网——每日最新资讯28at.com

与 C 对比

上面的示例中,我们使用 os.Stat 函数获取文件的状态,通过 errors.Is 判断返回错误,如果是 os.ErrNotExist,则文件不存在。8BX28资讯网——每日最新资讯28at.com

不得不说,这其实更底层更标准的做法。8BX28资讯网——每日最新资讯28at.com

类似于 Python 等高级语言,提供 os.path.exist 主要是为了方便编程,提高效率。8BX28资讯网——每日最新资讯28at.com

如果使用 Unix C 实现同样的功能,示例代码如下:8BX28资讯网——每日最新资讯28at.com

#include <errno.h>#include <stdio.h>#include <sys/stat.h>int main() {  struct stat buffer;  int exist = stat("/path/to/file", &buffer);  if (exist != 0) {    if (errno == ENOENT) { /* 文件不存在*/ }     else { /* 其他错误 */ }    return 0;  }  // 文件存在  return 0;}

是不是和我们前面代码基本是一个模子。8BX28资讯网——每日最新资讯28at.com

Go1.13 以及之后推荐使用 errors.Is

自 Go 1.13 起,推荐使用 os.Stat 和 errors.Is 的组合。这种方法提供了更一致和灵活的错误处理方式。8BX28资讯网——每日最新资讯28at.com

具体而言,即使是经过包裹的错误,errors.Is 依然能够识别。8BX28资讯网——每日最新资讯28at.com

我期初认为,os.IsNotExist 能识别包裹 error,但不太确定,于是写了个代码简单测试了下。8BX28资讯网——每日最新资讯28at.com

示例代码,如下所示:8BX28资讯网——每日最新资讯28at.com

_, err := os.Stat("/path/to/file")  // 这是一个不存在的文件路径werr := fmt.Errorf("Main: %w", err) // 包裹生成新错误fmt.Println(os.IsNotExist(err))     // 返回 true,表示不存在,这是错误结果fmt.Println(os.IsNotExist(werr))    // 返回 false,表示存在fmt.Println(errors.Is(werr, os.ErrNotExist)) // 返回 true 表示不存在

测试结果都已写在注释中。8BX28资讯网——每日最新资讯28at.com

如上可知, os.IsNotExist 只能识别最初的 error,如果错误经过 fmt.Errorf 包裹,则必须使用 errors.Is 识别。8BX28资讯网——每日最新资讯28at.com

图片图片8BX28资讯网——每日最新资讯28at.com

一句话概括,os.IsNotExist 可以用,但有适用范围,而 errors.Is 则更通用。8BX28资讯网——每日最新资讯28at.com

这一般也同样适用于其他类似的库。8BX28资讯网——每日最新资讯28at.com

直接使用 Open 避免竞态条件

到这里,基本已经解答了 Go 中如何检查文件存在性的问题。8BX28资讯网——每日最新资讯28at.com

但,我还想引入一个讨论:并发场景下,如何避免检查文件存在性时引入潜在的竞态条件?8BX28资讯网——每日最新资讯28at.com

简言之,文件状态可能在检查和操作发生变化。8BX28资讯网——每日最新资讯28at.com

图片图片8BX28资讯网——每日最新资讯28at.com

什么是更好的做法呢?8BX28资讯网——每日最新资讯28at.com

我们可以直接尝试打开或操作文件,根据返回结果判断错误。8BX28资讯网——每日最新资讯28at.com

示例代码如下:8BX28资讯网——每日最新资讯28at.com

file, err := os.Open("/path/to/file")if err != nil {    if errors.Is(err, os.ErrNotExist) {        // 文件不存在    } else {        // 处理其他类型的错误    }}

如上代码中,你通过 open 直接打开一个文件,如果文件不存在,os.Open 将返回一个错误,我们检查 error 确定下一步的操作。8BX28资讯网——每日最新资讯28at.com

通过这种方式,我们可以避免打开文件时引入竞态条件。8BX28资讯网——每日最新资讯28at.com

open 是原子操作?

读到这里,可能有人不禁问,为什么 open 能避免竞态条件呢?它是原子操作吗?8BX28资讯网——每日最新资讯28at.com

是的。8BX28资讯网——每日最新资讯28at.com

图片图片8BX28资讯网——每日最新资讯28at.com

系统调用都是原子操作,操作系统会保证操作过程不受到干扰。如果出现问题,也会进行回滚操作.8BX28资讯网——每日最新资讯28at.com

这一点对于 Open 同样使用。8BX28资讯网——每日最新资讯28at.com

当我们使用 open 打开一个文件时,系统会确保在这个操作完成前,不会受其他操作干扰,包括如检查文件是否存在、创建文件描述符、分配必要的资源等。8BX28资讯网——每日最新资讯28at.com

结论

本文通过一个小小的问题:Go 语言中如何检查文件是否存在,除了引出 Go 中检查文件是否存在的基本方法。同时,还介绍了文件操作时如何避免潜在的竞态条件,进一步了解到一个有趣的小知识,Unix 系统调用是原子性操作。8BX28资讯网——每日最新资讯28at.com

最后,还是希望本文能帮助各位在 GO 语言的学习道路上起到一点微末作用。8BX28资讯网——每日最新资讯28at.com

博客地址:Go 中如何检查文件是否存在?可能产生竞态条件?[1]8BX28资讯网——每日最新资讯28at.com

引用链接

[1] Go 中如何检查文件是否存在?可能产生竞态条件?: https://www.poloxue.com/posts/2024-02-05-check-if-file-exists-in-golang/8BX28资讯网——每日最新资讯28at.com

8BX28资讯网——每日最新资讯28at.com

本文链接:http://www.28at.com/showinfo-26-74190-0.htmlGo 中如何检查文件是否存在?可能产生竞态条件?

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: ​Thread Local深度解析,你学会了吗?

下一篇: 100000行级别数据的 Excel 导入优化之路

标签:
  • 热门焦点
  • 卢伟冰长文解析K60至尊版 对Redmi有着里程碑式的意义

    在今天的Redmi后性能时代战略发布会结束之后,Redmi总经理卢伟冰又带来了一篇长文,详解了为什么 Redmi 要开启后性能时代?为什么选择和 MediaTek、Pixelworks 深度合作?以及后性
  • 俄罗斯:将审查iPhone等外国公司设备 保数据安全

    iPhone和特斯拉都属于在各自领域领头羊的品牌,推出的产品也也都是数一数二的,但对于一些国家而言,它们的产品可靠性和安全性还是在限制范围内。近日,俄罗斯联邦通信、信息技术
  • 红魔电竞平板评测:大屏幕硬实力

    前言:三年的疫情因为要上网课的原因激活了平板市场,如今网课的时代已经过去,大家的生活都恢复到了正轨,这也就意味着,真正考验平板电脑生存的环境来了。也就是面对着这种残酷的
  • Golang 中的 io 包详解:组合接口

    io.ReadWriter// ReadWriter is the interface that groups the basic Read and Write methods.type ReadWriter interface { Reader Writer}是对Reader和Writer接口的组合,
  • 三言两语说透柯里化和反柯里化

    JavaScript中的柯里化(Currying)和反柯里化(Uncurrying)是两种很有用的技术,可以帮助我们写出更加优雅、泛用的函数。本文将首先介绍柯里化和反柯里化的概念、实现原理和应用
  • 自律,给不了Keep自由!

    来源 | 互联网品牌官作者 | 李大为编排 | 又耳 审核 | 谷晓辉自律能不能给用户自由暂时不好说,但大概率不能给Keep自由。近日,全球最大的在线健身平台Keep正式登陆港交所,努力
  • 网传小米汽车开始筛选交付中心 建筑面积不低于3000平方米

    7月7日消息,近日有微博网友@长三角行健者爆料称,据经销商集团反馈,小米汽车目前已经开始了交付中心的筛选工作,要求候选场地至少有120个车位,建筑不能低
  • iQOO Neo8 Pro评测:旗舰双芯加持 最强性能游戏旗舰

    【Techweb评测】去年10月,iQOO推出了一款Neo7手机,该机搭载了联发科天玑9000+,配备独显芯片Pro+,带来了同价位段最佳的游戏体验,一经上市便受到了诸多用
  • 英特尔Xe-HP项目终止,将专注Xe-HPC/HPG系列显卡

    据10 月 31 日消息报道,英特尔高级副总裁兼加速计算系统和图形事业部总经理 表示,Xe-HP“ Arctic Sound” 系列服务器 GPU 已经应用于 oneAPI devcloud 云服
Top