- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章Go语言并发编程 互斥锁详情由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
Go 语言的同步工具主要由 sync 包提供,互斥锁 (Mutex) 与读写锁 (RWMutex) 就是sync 包中的方法.
互斥锁可以用来保护一个临界区,保证同一时刻只有一个 goroutine 处于该临界区内。主要包括锁定(Lock方法)和解锁(Unlock方法)两个操作,首先对进入临界区的goroutine进行锁定,离开时进行解锁.
使用互斥锁 (Mutex)时要注意以下几点:
deadlock
);sync.Mutex
类型属于值类型,将它传给一个函数时,会产生一个副本,在函数中对锁的操作不会影响原锁总之,一个互斥锁只用来保护一个临界区,加锁后记得解锁,对于每一个锁定操作,都要有且只有一个对应的解锁操作,也就是加锁和解锁要成对出现,最保险的做法时使用defer语句解锁.
下面的代码模拟取钱和存钱操作:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
|
package main
import
(
"flag"
"fmt"
"sync"
)
var (
mutex
sync
.Mutex
balance int
protecting uint
//
是否加锁
sign =
make
(chan struct{}, 10)
//
通道,用于等待所有goroutine
)
//
存钱
func deposit(value int) {
defer func() {
sign <- struct{}{}
}()
if
protecting == 1 {
mutex.Lock()
defer mutex.Unlock()
}
fmt
.Printf(
"余额: %d\n"
, balance)
balance += value
fmt
.Printf(
"存 %d 后的余额: %d\n"
, value, balance)
fmt
.Println()
}
//
取钱
func withdraw(value int) {
defer func() {
sign <- struct{}{}
}()
if
protecting == 1 {
mutex.Lock()
defer mutex.Unlock()
}
fmt
.Printf(
"余额: %d\n"
, balance)
balance -= value
fmt
.Printf(
"取 %d 后的余额: %d\n"
, value, balance)
fmt
.Println()
}
func main() {
for
i:=0; i < 5; i++ {
go withdraw(500)
//
取500
go deposit(500)
//
存500
}
for
i := 0; i < 10; i++ {
<-sign
}
fmt
.Printf(
"当前余额: %d\n"
, balance)
}
func init() {
balance = 1000
//
初始账户余额为1000
flag.UintVar(&protecting,
"protecting"
, 0,
"是否加锁,0表示不加锁,1表示加锁"
)
}
|
上面的代码中,使用了通道来让主 goroutine 等待其他 goroutine 运行结束,每个子goroutine在运行结束之前向通道发送一个元素,主 goroutine 在最后从这个通道接收元素,接收次数与子goroutine个数相同。接收完后就会退出主goroutine.
代码使用协程实现多次(5次)对一个账户进行存钱和取钱的操作,先来看不加锁的情况 。
余额: 1000 存 500 后的余额: 1500 余额: 1000 取 500 后的余额: 1000 余额: 1000 存 500 后的余额: 1500 余额: 1000 取 500 后的余额: 1000 余额: 1000 存 500 后的余额: 1500 余额: 1000 取 500 后的余额: 1000 余额: 1000 取 500 后的余额: 500 余额: 1000 存 500 后的余额: 1000 余额: 1000 取 500 后的余额: 500 余额: 1000 存 500 后的余额: 1000 当前余额: 1000 。
可以看到出现了混乱,比如第二次1000的余额取500后还是1000,这种对同一资源的竞争出现了竞态条件(Race Condition).
下面来看加锁的执行结果:
余额: 1000 取 500 后的余额: 500 。
余额: 500 存 500 后的余额: 1000 。
余额: 1000 取 500 后的余额: 500 。
余额: 500 存 500 后的余额: 1000 。
余额: 1000 取 500 后的余额: 500 。
余额: 500 存 500 后的余额: 1000 。
余额: 1000 存 500 后的余额: 1500 。
余额: 1500 取 500 后的余额: 1000 。
余额: 1000 取 500 后的余额: 500 。
余额: 500 存 500 后的余额: 1000 。
当前余额: 1000 加锁后就正常了.
下面介绍更细化的互斥锁:读/写互斥锁RWMutex.
读/写互斥锁RWMutex包含了读锁和写锁,分别对共享资源的“读操作”和“写操作”进行保护。sync.RWMutex类型中的Lock方法和Unlock方法分别用于对写锁进行锁定和解锁,而它的RLock方法和RUnlock方法则分别用于对读锁进行锁定和解锁.
有了互斥锁Mutex,为什么还需要读写锁呢?因为在很多并发操作中,并发读取占比很大,写操作相对较少,读写锁可以并发读取,这样可以提供服务性能。读写锁具有以下特征:
。
读写锁 | 读锁 | 写锁 |
---|---|---|
读锁 | Yes | No |
写锁 | No | No |
。
也就是说:
goroutine
不能进行写操作。换句话说就是读写操作和写写操作不能并行执行,也就是读写互斥;在使用读写锁时,还需要注意:
改写前面的取钱和存钱操作,添加查询余额的方法:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
|
package main
import
(
"fmt"
"sync"
)
//
account 代表计数器。
type
account struct {
num uint
//
操作次数
balance int
//
余额
rwMu *
sync
.RWMutex
//
读写锁
}
var sign =
make
(chan struct{}, 15)
//
通道,用于等待所有goroutine
//
查看余额:使用读锁
func (c *account) check() {
defer func() {
sign <- struct{}{}
}()
c.rwMu.RLock()
defer c.rwMu.RUnlock()
fmt
.Printf(
"%d 次操作后的余额: %d\n"
, c.num, c.balance)
}
//
存钱:写锁
func (c *account) deposit(value int) {
defer func() {
sign <- struct{}{}
}()
c.rwMu.Lock()
defer c.rwMu.Unlock()
fmt
.Printf(
"余额: %d\n"
, c.balance)
c.num += 1
c.balance += value
fmt
.Printf(
"存 %d 后的余额: %d\n"
, value, c.balance)
fmt
.Println()
}
//
取钱:写锁
func (c *account) withdraw(value int) {
defer func() {
sign <- struct{}{}
}()
c.rwMu.Lock()
defer c.rwMu.Unlock()
fmt
.Printf(
"余额: %d\n"
, c.balance)
c.num += 1
c.balance -= value
fmt
.Printf(
"取 %d 后的余额: %d\n"
, value, c.balance)
fmt
.Println()
}
func main() {
c := account{0, 1000, new(
sync
.RWMutex)}
for
i:=0; i < 5; i++ {
go c.withdraw(500)
//
取500
go c.deposit(500)
//
存500
go c.check()
}
for
i := 0; i < 15; i++ {
<-sign
}
fmt
.Printf(
"%d 次操作后的余额: %d\n"
, c.num, c.balance)
}
|
执行结果:
余额: 1000 取 500 后的余额: 500 。
1 次操作后的余额: 500 1 次操作后的余额: 500 1 次操作后的余额: 500 1 次操作后的余额: 500 1 次操作后的余额: 500 余额: 500 存 500 后的余额: 1000 。
余额: 1000 取 500 后的余额: 500 。
余额: 500 存 500 后的余额: 1000 。
余额: 1000 存 500 后的余额: 1500 。
余额: 1500 取 500 后的余额: 1000 。
余额: 1000 取 500 后的余额: 500 。
余额: 500 存 500 后的余额: 1000 。
余额: 1000 取 500 后的余额: 500 。
余额: 500 存 500 后的余额: 1000 。
10 次操作后的余额: 1000 。
读写锁和互斥锁的不同之处在于读写锁把对共享资源的读操作和写操作分开了,可以实现更复杂的访问控制.
总结:
读写锁也是一种互斥锁,它是互斥锁的扩展。在使用时需要注意:
到此这篇关于Go语言并发编程 互斥锁详情的文章就介绍到这了,更多相关Go语言并发编程 互斥锁内容请搜索我以前的文章或继续浏览下面的相关文章希望大家以后多多支持我! 。
原文链接:https://blog.csdn.net/u010698107/article/details/120248679 。
最后此篇关于Go语言并发编程 互斥锁详情的文章就讲到这里了,如果你想了解更多关于Go语言并发编程 互斥锁详情的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
至少在某些 ML 系列语言中,您可以定义可以执行模式匹配的记录,例如http://learnyouahaskell.com/making-our-own-types-and-typeclasses -
这可能是其他人已经看到的一个问题,但我正在尝试寻找一种专为(或支持)并发编程而设计的语言,该语言可以在 .net 平台上运行。 我一直在 erlang 中进行辅助开发,以了解该语言,并且喜欢建立一个稳
As it currently stands, this question is not a good fit for our Q&A format. We expect answers to be
我正在寻找一种进程间通信工具,可以在相同或不同系统上运行的语言和/或环境之间使用。例如,它应该允许在 Java、C# 和/或 C++ 组件之间发送信号,并且还应该支持某种排队机制。唯一明显与环境和语言
我有一些以不同语言返回的文本。现在,客户端返回的文本格式为(en-us,又名美国英语): Stuff here to keep. -- Delete Here -- all of this below
问题:我希望在 R 中找到类似 findInterval 的函数,它为输入提供一个标量和一个表示区间起点的向量,并返回标量落入的区间的索引。例如在 R 中: findInterval(x = 2.6,
我是安卓新手。我正在尝试进行简单的登录 Activity ,但当我单击“登录”按钮时出现运行时错误。我认为我没有正确获取数据。我已经检查过,SQLite 中有一个与该 PK 相对应的数据。 日志猫。
大家好,感谢您帮助我。 我用 C# 制作了这个计算器,但遇到了一个问题。 当我添加像 5+5+5 这样的东西时,它给了我正确的结果,但是当我想减去两个以上的数字并且还想除或乘以两个以上的数字时,我没有
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 4 年前。 Improve th
这就是我所拥有的 #include #include void print(int a[], int size); void sort (int a[], int size); v
你好,我正在寻找我哪里做错了? #include #include int main(int argc, char *argv[]) { int account_on_the_ban
嘿,当我开始向数组输入数据时,我的代码崩溃了。该程序应该将数字读入数组,然后将新数字插入数组中,最后按升序排列所有内容。我不确定它出了什么问题。有人有建议吗? 这是我的代码 #include #in
我已经盯着这个问题好几个星期了,但我一无所获!它不起作用,我知道那么多,但我不知道为什么或出了什么问题。我确实知道开发人员针对我突出显示的行吐出了“错误:预期表达式”,但这实际上只是冰山一角。如果有人
我正在编写一个点对点聊天程序。在此程序中,客户端和服务器功能写入一个唯一的文件中。首先我想问一下我程序中的机制是否正确? I fork() two processes, one for client
基本上我需要找到一种方法来发现段落是否以句点 (.) 结束。 此时我已经可以计算给定文本的段落数,但我没有想出任何东西来检查它是否在句点内结束。 任何帮助都会帮助我,谢谢 char ch; FI
我的函数 save_words 接收 Armazena 和大小。 Armazena 是一个包含段落的动态数组,size 是数组的大小。在这个函数中,我想将单词放入其他称为单词的动态数组中。当我运行它时
我有一个结构 struct Human { char *name; struct location *location; int
我正在尝试缩进以下代码的字符串输出,但由于某种原因,我的变量不断从文件中提取,并且具有不同长度的噪声或空间(我不确定)。 这是我的代码: #include #include int main (v
我想让用户选择一个选项。所以我声明了一个名为 Choice 的变量,我希望它输入一个只能是 'M' 的 char 、'C'、'O' 或 'P'。 这是我的代码: char Choice; printf
我正在寻找一种解决方案,将定义和变量的值连接到数组中。我已经尝试过像这样使用 memcpy 但它不起作用: #define ADDRESS {0x00, 0x00, 0x00, 0x00, 0x0
我是一名优秀的程序员,十分优秀!