gmutex.Mutex mutex objects support read/write control, with functionality similar to the standard library’s sync.RWMutex, allowing concurrent reads but not concurrent writes.

For design details of the mutex, it’s recommended to read the lightweight high-definition version of the implementation source code: https://github.com/gogf/gf/blob/master/os/gmutex/gmutex.go

Usage:

  1. import "github.com/gogf/gf/v2/os/gmutex"

Interface Documentation:

https://pkg.go.dev/github.com/gogf/gf/v2/os/gmutex

  1. type Mutex
  2. func (m *Mutex) LockFunc(f func())
  3. func (m *Mutex) TryLockFunc(f func()) (result bool)
  4. type RWMutex
  5. func New() *RWMutex
  6. func (m *RWMutex) LockFunc(f func())
  7. func (m *RWMutex) RLockFunc(f func())
  8. func (m *RWMutex) TryLockFunc(f func()) (result bool)
  9. func (m *RWMutex) TryRLockFunc(f func()) (result bool)
  1. The standout feature of this mutex module is the support for Try* methods and *Func methods.
  2. Try* methods are used to attempt to acquire a specific type of lock. If the lock is acquired successfully, it immediately returns true; otherwise, it returns false without blocking, which is very practical for business logic requiring non-blocking lock mechanisms.
  3. *Func methods use closure anonymous functions for concurrent safety lock control in specific scopes, which is particularly convenient for concurrent safety control of specific code blocks. As it uses defer internally to release locks, even if exceptions or errors occur within the function, it will not affect the safety control of the locking mechanism.

Benchmark

Benchmark comparison results of gmutex.Mutex with the standard library’s sync.Mutex and sync.RWMutex: gmutex_bench_test.go

  1. goos: linux
  2. goarch: amd64
  3. pkg: github.com/gogf/gf/v2/os/gmutex
  4. Benchmark_Mutex_LockUnlock-4 50000000 31.5 ns/op
  5. Benchmark_RWMutex_LockUnlock-4 30000000 54.1 ns/op
  6. Benchmark_RWMutex_RLockRUnlock-4 50000000 27.9 ns/op
  7. Benchmark_GMutex_LockUnlock-4 50000000 27.2 ns/op
  8. Benchmark_GMutex_TryLock-4 100000000 16.7 ns/op
  9. Benchmark_GMutex_RLockRUnlock-4 50000000 38.0 ns/op
  10. Benchmark_GMutex_TryRLock-4 100000000 16.8 ns/op

Example 1, Basic Usage

  1. package main
  2. import (
  3. "time"
  4. "github.com/gogf/gf/v2/os/gctx"
  5. "github.com/gogf/gf/v2/os/glog"
  6. "github.com/gogf/gf/v2/os/gmutex"
  7. )
  8. func main() {
  9. ctx := gctx.New()
  10. mu := gmutex.New()
  11. for i := 0; i < 10; i++ {
  12. go func(n int) {
  13. mu.Lock()
  14. defer mu.Unlock()
  15. glog.Print(ctx, "Lock:", n)
  16. time.Sleep(time.Second)
  17. }(i)
  18. }
  19. for i := 0; i < 10; i++ {
  20. go func(n int) {
  21. mu.RLock()
  22. defer mu.RUnlock()
  23. glog.Print(ctx, "RLock:", n)
  24. time.Sleep(time.Second)
  25. }(i)
  26. }
  27. time.Sleep(15 * time.Second)
  28. }

After execution, the terminal output:

  1. 2019-07-13 16:19:55.417 Lock: 0
  2. 2019-07-13 16:19:56.421 Lock: 1
  3. 2019-07-13 16:19:57.424 RLock: 0
  4. 2019-07-13 16:19:57.424 RLock: 4
  5. 2019-07-13 16:19:57.425 RLock: 8
  6. 2019-07-13 16:19:57.425 RLock: 2
  7. 2019-07-13 16:19:57.425 RLock: 7
  8. 2019-07-13 16:19:57.425 RLock: 5
  9. 2019-07-13 16:19:57.425 RLock: 9
  10. 2019-07-13 16:19:57.425 RLock: 1
  11. 2019-07-13 16:19:57.425 RLock: 6
  12. 2019-07-13 16:19:57.425 RLock: 3
  13. 2019-07-13 16:19:58.429 Lock: 3
  14. 2019-07-13 16:19:59.433 Lock: 4
  15. 2019-07-13 16:20:00.438 Lock: 5
  16. 2019-07-13 16:20:01.443 Lock: 6
  17. 2019-07-13 16:20:02.448 Lock: 7
  18. 2019-07-13 16:20:03.452 Lock: 8
  19. 2019-07-13 16:20:04.456 Lock: 9
  20. 2019-07-13 16:20:05.461 Lock: 2

The purpose of using glog for printing is to conveniently observe the print output time. You can see that in the third second, the read lock seized the opportunity. Since gmutex.Mutex objects support concurrent reads but not concurrent writes, the read lock quickly executed after seizing; meanwhile, the write lock continues executing with one log per second.

Example 2, *Func Usage

  1. package main
  2. import (
  3. "time"
  4. "github.com/gogf/gf/v2/os/glog"
  5. "github.com/gogf/gf/v2/os/gmutex"
  6. )
  7. func main() {
  8. mu := gmutex.New()
  9. go mu.LockFunc(func() {
  10. glog.Println("lock func1")
  11. time.Sleep(1 * time.Second)
  12. })
  13. time.Sleep(time.Millisecond)
  14. go mu.LockFunc(func() {
  15. glog.Println("lock func2")
  16. })
  17. time.Sleep(2 * time.Second)
  18. }

After execution, the terminal output:

  1. 2019-07-13 16:28:10.381 lock func1
  2. 2019-07-13 16:28:11.385 lock func2

As you can see, using *Func methods for lock control in specific scopes is very convenient.