Home >Backend Development >Golang >Practical sharing of synchronization and lock protection of Golang functions

Practical sharing of synchronization and lock protection of Golang functions

WBOY
WBOYOriginal
2023-05-16 09:04:571692browse

With the development of the Internet and the popularization of cloud computing and big data technology, modern software systems need to process more and more data, and they also need to ensure the efficiency and reliability of the system. In this context, the performance and technical features of the language become particularly important. Among them, Golang, as an efficient, lightweight, and highly concurrency programming language, has received more and more attention and application in recent years. This article will discuss the synchronization and lock protection practices of Golang functions, and provide some useful experience sharing for Golang developers.

  1. Principles and methods of synchronization

Synchronization is the key to collaboration between multiple threads or processes. Its main purpose is to ensure the correct access and protection of various resources. . In Golang, the main means of realizing synchronization are as follows:

1.1 Mutex lock (sync.Mutex)

Mutex lock is the most basic synchronization mechanism in Golang. Its main role is to ensure that only one goroutine can access shared resources at the same time. When a goroutine requests the resource, it will try to acquire the lock. If it cannot obtain it, it will be blocked until the lock is released. The following is an example of a simple mutex implementation:

package main

import (
    "fmt"
    "sync"
)

var count int
var mu sync.Mutex // 互斥锁

func main() {
    for i := 0; i < 10; i++ {
        go increase()
    }

    // 等待所有goroutine执行完成
    for {
        mu.Lock()
        if count == 10 {
            mu.Unlock()
            break
        }
        mu.Unlock()
    }

    fmt.Println("count:", count)
}

func increase() {
    mu.Lock()
    defer mu.Unlock()
    count += 1
}

In the above example, we use a mutex to ensure atomic operation of the shared variable count. Inside the increase function, we first acquire the mutex lock, then perform an increment operation on count, and finally release the lock. In this way, we can prevent concurrent access to count from causing unexpected results.

1.2 Read-write lock (sync.RWMutex)

RWMutex is an advanced mutex lock that supports multiple read operations concurrently, but only allows one write operation. In implementation, it organizes the read operations of multiple goroutines by switching between read and write modes, thereby improving concurrency performance. The following is an example of a simple read-write lock implementation:

package main

import (
    "fmt"
    "sync"
)

var count int
var mu sync.RWMutex // 读写锁

func main() {
    for i := 0; i < 10; i++ {
        go increase()
    }

    // 等待所有goroutine执行完成
    for {
        mu.RLock()
        if count == 10 {
            mu.RUnlock()
            break
        }
        mu.RUnlock()
    }

    fmt.Println("count:", count)
}

func increase() {
    mu.Lock()
    defer mu.Unlock()
    count += 1
}

In the above example, we use read-write locks to ensure atomic operations of the shared variable count. Inside the increase function, we first acquire the write lock of the read-write lock, then perform an increment operation on count, and finally release the lock. In this way, we can prevent concurrent access to count from causing unexpected results.

  1. Practice of lock protection

In addition to the synchronization mechanism, Golang also provides some practical methods of lock protection to ensure the integrity and security of data. The following is a detailed introduction to some practical methods:

2.1 Atomic operation (sync/atomic)

Atomic operation is a technology that can ensure data synchronization without locking. Golang provides a series of atomic operation functions to implement basic memory synchronization functions. The following is an example:

package main

import (
    "fmt"
    "sync/atomic"
)

var count int32

func main() {
    for i := 0; i < 10; i++ {
        go increase()
    }

    // 等待所有goroutine执行完成
    for {
        if atomic.LoadInt32(&count) == 10 {
            break
        }
    }

    fmt.Println("count:", count)
}

func increase() {
    atomic.AddInt32(&count, 1)
}

In the above example, we use the atomic operation function atomic.AddInt32() to ensure that the increment operation of count is atomic, thereby avoiding data corruption caused by race conditions. abnormal.

2.2 Channel Communication

Channel is an important synchronization tool in Golang. It ensures the correctness of data through communication between goroutines. Channel is somewhat similar to a Unix pipe, which allows one goroutine to send a data block to another goroutine, or receive a data block. The following is an example:

package main

import (
    "fmt"
)

func main() {
    ch := make(chan int)
    go increase(ch)

    // 接收所有增加的值
    count := 0
    for i := 0; i < 10; i++ {
       count += <-ch
    }

    fmt.Println("count:", count)
}

func increase(ch chan int) {
    for i := 0; i < 10; i++ {
       ch <- 1
    }
    close(ch)
}

In the above example, we use channels to prevent race conditions caused by the shared data count being accessed concurrently by multiple goroutines. Inside the increase function, we send 10 1s to the main function through the channel to perform the counting operation. Inside the main function, we receive the data in the channel through a loop and accumulate it into the count variable, thus avoiding data exceptions caused by race conditions.

2.3 The defer statement of sync.Mutex

In Golang, mutex locks often use the defer statement to ensure the correct release of the lock. The defer statement is a mechanism that causes the statement to be executed when the function returns. It can avoid program exceptions caused by forgetting to release the lock. The following is an example:

package main

import (
    "fmt"
    "sync"
)

var count int
var mu sync.Mutex // 互斥锁

func main() {
    for i := 0; i < 10; i++ {
        go increase()
    }

    // 等待所有goroutine执行完成
    for {
        mu.Lock()
        if count == 10 {
            mu.Unlock()
            break
        }
        mu.Unlock()
    }

    fmt.Println("count:", count)
}

func increase() {
    mu.Lock()
    defer mu.Unlock()
    count += 1
}

In the above example, we use the defer statement to ensure the correct release of the mutex lock. When the goroutine leaves the increase function, the defer statement will automatically release the lock to ensure that the next time the lock is acquired, it can be executed successfully.

Conclusion

The above is the practical sharing of synchronization and lock protection of Golang functions. Through the application of mutex locks, read-write locks, atomic operations, Channel communication, and defer statements, we can better ensure the correctness and security of data in Golang multi-threaded programming. Whether in large-scale cloud computing systems, distributed systems or real-time data processing systems, these synchronization and lock protection technologies are of great significance.

The above is the detailed content of Practical sharing of synchronization and lock protection of Golang functions. For more information, please follow other related articles on the PHP Chinese website!

Statement:
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn