中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

golang如何實現文件鎖

發布時間:2022-12-27 13:48:17 來源:億速云 閱讀:127 作者:iii 欄目:編程語言

本篇內容主要講解“golang如何實現文件鎖”,感興趣的朋友不妨來看看。本文介紹的方法操作簡單快捷,實用性強。下面就讓小編來帶大家學習“golang如何實現文件鎖”吧!

在golang中,可以利用sync包的api來實現文件鎖。文件鎖(flock)是對于整個文件的建議性鎖;也就是說,如果一個進程在一個文件(inode)上放了鎖,其它進程是可以知道的(建議性鎖不強求進程遵守);文件鎖的調用語法“syscall.Flock(int(f.Fd()), syscall.LOCK_EX|syscall.LOCK_NB)”。

我們使用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如何實現文件鎖”有了更深的了解,不妨來實際操作一番吧!這里是億速云網站,更多相關內容可以進入相關頻道進行查詢,關注我們,繼續學習!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

招远市| 淮北市| 义乌市| 西和县| 玛纳斯县| 东乡族自治县| 宽城| 东兰县| 荣成市| 阜康市| 汨罗市| 石林| 德惠市| 马公市| 定远县| 娱乐| 屏南县| 龙州县| 高台县| 兰西县| 卢氏县| 新余市| 长垣县| 扶沟县| 上饶市| 阿荣旗| 固安县| 温泉县| 泸定县| 宁津县| 临汾市| 榆中县| 巴青县| 贵溪市| 龙里县| 哈尔滨市| 大庆市| 留坝县| 泰州市| 株洲市| 合水县|