首頁 後端開發 Golang golang怎麼實作文件鎖

golang怎麼實作文件鎖

Dec 19, 2022 am 11:03 AM
golang go語言

在golang中,可以利用sync套件的api來實作檔案鎖定。文件鎖(flock)是對於整個文件的建議性鎖;也就是說,如果一個進程在一個文件(inode)上放了鎖,其它進程是可以知道的(建議性鎖不強求進程遵守);文件鎖的呼叫語法「syscall.Flock(int(f.Fd()), syscall.LOCK_EX|syscall.LOCK_NB)」。

golang怎麼實作文件鎖

本教學操作環境:windows7系統、GO 1.18版本、Dell G3電腦。

我們使用Go語言開發一些程式的時候,往往出現多個行程同時操作同一份文件的情況,這很容易導致文件中的資料混亂。這時我們就需要採取一些手段來平衡這些衝突,文件鎖(flock)應運而生,下面我們就來介紹一下。

對於flock,最常見的例子就是Nginx,進程運行起來後就會把當前的PID 寫入這個文件,當然如果這個文件已經存在了,也就是前一個進程還沒有退出,那麼Nginx就不會重新啟動,所以flock 還可以用來偵測進程是否存在。

flock 是整個檔案的建議性鎖定。也就是說,如果一個進程在一個檔案(inode)上放了鎖,其它進程是可以知道的(建議性鎖不強求進程遵守)。最棒的一點是,它的第一個參數是檔案描述符,在此檔案描述符關閉時,鎖定會自動釋放。而當進程終止時,所有的檔案描述符都會關閉。所以很多時候就不用考慮類似原子鎖解鎖的事情。

在具體介紹前,先上程式碼

package main
import (
    "fmt"
    "os"
    "sync"
    "syscall"
    "time"
)
//文件锁
type FileLock struct {
    dir string
    f   *os.File
}
func New(dir string) *FileLock {
    return &FileLock{
        dir: dir,
    }
}
//加锁
func (l *FileLock) Lock() error {
    f, err := os.Open(l.dir)
    if err != nil {
        return err
    }
    l.f = f
    err = syscall.Flock(int(f.Fd()), syscall.LOCK_EX|syscall.LOCK_NB)
    if err != nil {
        return fmt.Errorf("cannot flock directory %s - %s", l.dir, err)
    }
    return nil
}
//释放锁
func (l *FileLock) Unlock() error {
    defer l.f.Close()
    return syscall.Flock(int(l.f.Fd()), syscall.LOCK_UN)
}
func main() {
    test_file_path, _ := os.Getwd()
    locked_file := test_file_path
    wg := sync.WaitGroup{}
    for i := 0; i < 10; i++ {
        wg.Add(1)
        go func(num int) {
            flock := New(locked_file)
            err := flock.Lock()
            if err != nil {
                wg.Done()
                fmt.Println(err.Error())
                return
            }
            fmt.Printf("output : %d\n", num)
            wg.Done()
        }(i)
    }
    wg.Wait()
    time.Sleep(2 * time.Second)
}
登入後複製

在Windows 系統下執行上面的程式碼會出現下面的錯誤:

golang怎麼實作文件鎖

##這是因為Windows 系統不支援pid 鎖,所以我們需要在Linux 或Mac 系統下才能正常執行上面的程式。

上面程式碼中示範了同時啟動 10 個 goroutinue,但在程式運行過程中,只有一個 goroutine 能獲得檔案鎖(flock)。其它的 goroutinue 在獲取不到 flock 後,會拋出異常的訊息。這樣即可達到同一檔案在指定的周期內只允許一個行程存取的效果。

程式碼中檔案鎖的具體呼叫:

syscall.Flock(int(f.Fd()), syscall.LOCK_EX|syscall.LOCK_NB)
登入後複製
我們採用了 syscall.LOCK_EX、syscall.LOCK_NB,這是什麼意思呢?

flock 屬於建議性鎖,不具備強制性。一個進程使用flock 將文件鎖住,另一個進程可以直接操作正在被鎖的文件,修改文件中的數據,原因在於flock 只是用來檢測文件是否被加鎖,針對文件已經被加鎖,另一個進程寫入資料的情況,核心不會阻止這個行程的寫入操作,也就是建議性鎖的核心處理策略。

flock 主要三種操作類型:

  • LOCK_SH:共享鎖,多個程序可以使用相同把鎖,常被用作讀取共享鎖;

  • LOCK_EX:排他鎖,同時只允許一個行程使用,常被用來當作寫鎖;

  • ##LOCK_UN:釋放鎖。
  • 程式使用 flock 嘗試鎖定檔案時,如果檔案已經被其他進程鎖住,則行程會被阻塞直到鎖被釋放掉,或是在呼叫 flock 的時候,採用 LOCK_NB 參數。在嘗試鎖住該文件的時候,發現已經被其他服務鎖住,會回傳錯誤,錯誤碼為 EWOULDBLOCK。

flock 鎖的釋放非常具有特色,即可呼叫LOCK_UN 參數來釋放檔案鎖,也可以透過關閉fd 的方式來釋放檔案鎖(flock 的第一個參數是fd),表示flock 會隨著進程的關閉而自動釋放掉。

更多程式相關知識,請造訪:

程式設計影片

! !

以上是golang怎麼實作文件鎖的詳細內容。更多資訊請關注PHP中文網其他相關文章!

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

Go語言中用於浮點數運算的庫有哪些? Go語言中用於浮點數運算的庫有哪些? Apr 02, 2025 pm 02:06 PM

Go語言中用於浮點數運算的庫介紹在Go語言(也稱為Golang)中,進行浮點數的加減乘除運算時,如何確保精度是�...

GoLand中自定義結構體標籤不顯示怎麼辦? GoLand中自定義結構體標籤不顯示怎麼辦? Apr 02, 2025 pm 05:09 PM

GoLand中自定義結構體標籤不顯示怎麼辦?在使用GoLand進行Go語言開發時,很多開發者會遇到自定義結構體標籤在�...

Go的爬蟲Colly中Queue線程的問題是什麼? Go的爬蟲Colly中Queue線程的問題是什麼? Apr 02, 2025 pm 02:09 PM

Go爬蟲Colly中的Queue線程問題探討在使用Go語言的Colly爬蟲庫時,開發者常常會遇到關於線程和請求隊列的問題。 �...

在Go語言中使用Redis Stream實現消息隊列時,如何解決user_id類型轉換問題? 在Go語言中使用Redis Stream實現消息隊列時,如何解決user_id類型轉換問題? Apr 02, 2025 pm 04:54 PM

Go語言中使用RedisStream實現消息隊列時類型轉換問題在使用Go語言與Redis...

在 Go 語言中,為什麼使用 Println 和 string() 函數打印字符串會出現不同的效果? 在 Go 語言中,為什麼使用 Println 和 string() 函數打印字符串會出現不同的效果? Apr 02, 2025 pm 02:03 PM

Go語言中字符串打印的區別:使用Println與string()函數的效果差異在Go...

在Go編程中,如何正確管理Mysql和Redis的連接與釋放資源? 在Go編程中,如何正確管理Mysql和Redis的連接與釋放資源? Apr 02, 2025 pm 05:03 PM

Go編程中的資源管理:Mysql和Redis的連接與釋放在學習Go編程過程中,如何正確管理資源,特別是與數據庫和緩存�...

Go語言中哪些庫是由大公司開發或知名的開源項目提供的? Go語言中哪些庫是由大公司開發或知名的開源項目提供的? Apr 02, 2025 pm 04:12 PM

Go語言中哪些庫是大公司開發或知名開源項目?在使用Go語言進行編程時,開發者常常會遇到一些常見的需求,�...

Golang的目的:建立高效且可擴展的系統 Golang的目的:建立高效且可擴展的系統 Apr 09, 2025 pm 05:17 PM

Go語言在構建高效且可擴展的系統中表現出色,其優勢包括:1.高性能:編譯成機器碼,運行速度快;2.並發編程:通過goroutines和channels簡化多任務處理;3.簡潔性:語法簡潔,降低學習和維護成本;4.跨平台:支持跨平台編譯,方便部署。

See all articles