golang的乐观锁与悲观锁

golang的乐观锁与悲观锁,第1张

golang的乐观与悲观锁 基本概念

基本概念

乐观锁和悲观锁是两种思想,用于解决并发场景下的数据竞争问题。

乐观锁:乐观锁在 *** 作数据时非常乐观,认为别人不会同时修改数据。因此乐观锁不会上锁,只是在执行更新的时候判断一下在此期间别人是否修改了数据:如果别人修改了数据则放弃 *** 作,否则执行 *** 作。

悲观锁:悲观锁在 *** 作数据时比较悲观,认为别人会同时修改数据。因此 *** 作数据时直接把数据锁住,直到 *** 作完成后才会释放锁;上锁期间其他人不能修改数据。

package main

import (
	"fmt"
	"sync"
	"sync/atomic"
	"time"
)

func IncreValue1() {
	value1++
}

func IncreValue2() {
	atomic.AddInt32(&value2, 1)
}

func IncreValue3() {
	lock.Lock()
	value3++
	lock.Unlock()
}

var value1, value2, value3 int32
var lock sync.Mutex

func main() {
	//开启1000个线程,并执行自增 *** 作
	for i := 0; i < 1000; i++ {
		go IncreValue1()
		go IncreValue2()
		go IncreValue3()
	}
	//打印结果
	time.Sleep(1000)
	fmt.Println("线程不安全:", value1)
	fmt.Println("乐观锁:", value2)
	fmt.Println("悲观锁:", value2)
}

value1的值每次执行都小于1000,因为没有做任何线程安全的 *** 作。
func AddInt32(addr *int32, delta int32) (new int32)

第一个参数是指针,是因为该函数需要获得到被 *** 作值在内存中的存放位置(如果不是指针就是传值了),以便施加特殊的CPU指令(原子性是在硬件层面保证的)。

从另一个角度看,对于一个不能被取址的数值,我们是无法进行原子 *** 作的。

atomic.AddInt64、atomic.AddUint32,atomic.AddUint64,atomic.AddUintptr除了支持这些类型外还有另外四种方法CompareAndSwapInt32,LoadInt32,StoreInt32,SwapInt32(比较并交换,载入(拿),存储(放),交换)

https://www.kancloud.cn/digest/batu-go/153537
http://ifeve.com/go-concurrency-atomic/

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/langs/995071.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-05-21
下一篇 2022-05-21

发表评论

登录后才能评论

评论列表(0条)

保存