如何在golang中檢查文件是否存在
Go 中如何檢查文件是否存在呢?
如果你用的是 Python,可通過 os.path.exists
這樣的標(biāo)準(zhǔn)庫函數(shù)實現(xiàn)。遺憾的是,Go 標(biāo)準(zhǔn)庫沒有提供這樣直接的函數(shù),但好在,沒有直接的,卻有不那么直接的方法。
本文將基于這個話題展開,介紹 Go 中如何檢查文件是否存在。
另外,本文最后還會介紹一個小注意點,即在判斷文件是否存在時,如何避免中潛在的競態(tài)條件。
os.Stat 檢查文件狀態(tài)
Go 標(biāo)準(zhǔn)庫雖然沒有提供類似于 os.Exist
這樣直接的函數(shù)檢查文件是否存在,但它提供另外一個函數(shù) os.Stat
。
os.Stat
函數(shù)的作用是獲取文件狀態(tài)信息,我們通過檢查它返回的錯誤即可知曉文件是否存在。
示例代碼,如下所示:
func main() { _, err := os.Stat("/path/to/file") if err != nil { if os.IsNotExist(err) { // 文件不存在 } else { // 其他錯誤 } } // 文件存在 }
第一個返回值表示文件信息,不是我們關(guān)心的重點,直接省略掉。
第二個返回值表示錯誤 error
。如果文件不存在,可通過檢查 os.IsNotExist
檢查 error
是否是 os.ErrNotExist
,確定文件是否存在。
與 C 對比
上面的示例中,我們使用 os.Stat
函數(shù)獲取文件的狀態(tài),通過 errors.Is
判斷返回錯誤,如果是 os.ErrNotExist
,則文件不存在。
不得不說,這其實更底層更標(biāo)準(zhǔn)的做法。
類似于 Python 等高級語言,提供 os.path.exist
主要是為了方便編程,提高效率。
如果使用 Unix C 實現(xiàn)同樣的功能,示例代碼如下:
#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; }
是不是和我們前面代碼基本是一個模子。
Go1.13 以及之后推薦使用 errors.Is
自 Go 1.13 起,推薦使用 os.Stat
和 errors.Is
的組合。這種方法提供了更一致和靈活的錯誤處理方式。
具體而言,即使是經(jīng)過包裹的錯誤,errors.Is
依然能夠識別。
我期初認為,os.IsNotExist
能識別包裹 error
,但不太確定,于是寫了個代碼簡單測試了下。
示例代碼,如下所示:
_, err := os.Stat("/path/to/file") // 這是一個不存在的文件路徑 werr := fmt.Errorf("Main: %w", err) // 包裹生成新錯誤 fmt.Println(os.IsNotExist(err)) // 返回 true,表示不存在,這是錯誤結(jié)果 fmt.Println(os.IsNotExist(werr)) // 返回 false,表示存在 fmt.Println(errors.Is(werr, os.ErrNotExist)) // 返回 true 表示不存在
測試結(jié)果都已寫在注釋中。
如上可知, os.IsNotExist
只能識別最初的 error
,如果錯誤經(jīng)過 fmt.Errorf
包裹,則必須使用 errors.Is
識別。
一句話概括,os.IsNotExist
可以用,但有適用范圍,而 errors.Is
則更通用。
這一般也同樣適用于其他類似的庫。
直接使用 Open 避免競態(tài)條件
到這里,基本已經(jīng)解答了 Go 中如何檢查文件存在性的問題。
但,我還想引入一個討論:并發(fā)場景下,如何避免檢查文件存在性時引入潛在的競態(tài)條件?
簡言之,文件狀態(tài)可能在檢查和操作發(fā)生變化。
什么是更好的做法呢?
我們可以直接嘗試打開或操作文件,根據(jù)返回結(jié)果判斷錯誤。
示例代碼如下:
file, err := os.Open("/path/to/file") if err != nil { if errors.Is(err, os.ErrNotExist) { // 文件不存在 } else { // 處理其他類型的錯誤 } }
如上代碼中,你通過 open
直接打開一個文件,如果文件不存在,os.Open
將返回一個錯誤,我們檢查 error 確定下一步的操作。
通過這種方式,我們可以避免打開文件時引入競態(tài)條件。
open 是原子操作
讀到這里,可能有人不禁問,為什么 open 能避免競態(tài)條件呢?它是原子操作嗎?
是的。
系統(tǒng)調(diào)用都是原子操作,操作系統(tǒng)會保證操作過程不受到干擾。如果出現(xiàn)問題,也會進行回滾操作.
這一點對于 Open 同樣使用。
當(dāng)我們使用 open 打開一個文件時,系統(tǒng)會確保在這個操作完成前,不會受其他操作干擾,包括如檢查文件是否存在、創(chuàng)建文件描述符、分配必要的資源等。
結(jié)論
本文通過一個小小的問題:Go 語言中如何檢查文件是否存在,除了引出 Go 中檢查文件是否存在的基本方法。同時,還介紹了文件操作時如何避免潛在的競態(tài)條件,進一步了解到一個有趣的小知識,Unix 系統(tǒng)調(diào)用是原子性操作。
到此這篇關(guān)于如何在golang中檢查文件是否存在的文章就介紹到這了,更多相關(guān)go檢查文件是否存在內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
淺談Golang中創(chuàng)建一個簡單的服務(wù)器的方法
這篇文章主要介紹了淺談Golang中創(chuàng)建一個簡單的服務(wù)器的方法,golang中的net/http包對網(wǎng)絡(luò)的支持非常好,這樣會讓我們比較容易的建立起一個相對簡單的服務(wù)器,有一定的參考價值,感興趣的小伙伴們可以參考一下2018-06-06golang框架gin的日志處理和zap lumberjack日志使用方式
這篇文章主要介紹了golang框架gin的日志處理和zap lumberjack日志使用方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-01-01go語言實現(xiàn)并發(fā)網(wǎng)絡(luò)爬蟲的示例代碼
本文主要介紹了go語言實現(xiàn)并發(fā)網(wǎng)絡(luò)爬蟲的示例代碼,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-03-03Go 自定義package包設(shè)置與導(dǎo)入操作
這篇文章主要介紹了Go 自定義package包設(shè)置與導(dǎo)入操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-05-05Go語言基礎(chǔ)函數(shù)包的使用學(xué)習(xí)
本文通過一個實現(xiàn)加減乘除運算的小程序來介紹go函數(shù)的使用,以及使用函數(shù)的注意事項,并引出了對包的了解和使用,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-05-05