race condition
Data Races vs Race Conditions - Cronokirby - https://cronokirby.com/posts/2019/06/data-races-vs-race-conditions/data-races-vs-race-conditions/
data race and race condition
race conditon
在并发的程序里
- 产生不确定行为,non-determinism
- 我们预期程序总是产生特定行为
1
2
3
4
5
6
7
8
9
10
|
func main() {
go func() {
for {
fmt.Println("Thread B")
}
}
for {
fmt.Println("Thread A")
}
}
|
如果我们期望按照 threadA-threadb 交替输出,则是race conditon
data race
- 多个线程同时访问一个变量
- 至少有一个是写入操作
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
未获得预期的结果
1. 多个线程并发执行, 访问了共享资源
2. 正确的结果需要依赖多个线程按照特定顺序执行.
3. 但是多线程并未按照特定的顺序执行, 线程执行的顺序不可预期的
1. In concurrent programming,
2. a race condition occurs when the correct behavior of a program depends on the order or timing of two or more threads' execution
3. and that order or timing is unpredictable.
type:
critial race conditon: 发生了race condition, 但是结果不影响程序的正确性
non-critical conditon: 发生了race conditon, 但影响程序正确性
non-critail race conditon
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
### data race
## problem:
what:
1. 一种同步机制;
2. 确保资源只能被一个线程
code:
|
acquire(loccker)
if get: go
if not get get: block
release
int flag = 0;
locker:
try set new value:
success: get locker
fail: wait
unlock:rest flag
flag=0
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
locker type:
1. optimistic locker:
2. check flag and do, check fail: rollback or return
3. no block other thread
```c
select version, name from user // tang,2
if success = [update name=jim where name =tang and version=2],!success{
rollback
}
compareandswap(flag,0, 1):
if flag==0: flag =1;
else return;
|
- pessimistic locker:
2. a: set flag and do, set fail: wait
3. block other thread
distributed lock
how:
- 获取标识
- 重置标识
1
2
3
|
compareandswap flag
unset flag
|
1
2
3
4
5
6
7
8
9
10
11
12
|
uuid
while(1):
result = SET locker uuiid NX EX 300
if result == nil: continue
else: break;
//.....
if get locker == uuid: del
else return;
|
key pointer:
- 设置过期时间(被动释放 ),防止锁无法释放
probem:
- 线程执行时间过长被动释放 ,另一线程获取锁
- 老线程释放新线程锁: add random value
- 在临界区有两个线程: watch dog 续期, redissession

- 集群问题: 非强一致性
- 主备切换时候
- 脑裂
etcd
how:
- try set min verison:
- set: revision = /locker/uuid and keepalive
- checkSuccess:
- success: the min revison of /locker/
- fail: wait pre key
- unlock: delete key
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
// compare and swap
leaseid = etcd lease grant 10
etcdctl lease keep-alive leaseid
reponse = put /locker+randid '' -- lease leaseid
// the min revision
list = etcdctl get --prefix=/locker --sort-by modify
if response.modversion = list[0].modversion:
get locker..,..
else
watch previous key
|
dead lock
what: 线程阻塞,等待锁的释放, 而由于循环等待, 锁永远不会被释放
example:
两个进程,互相持有对方需要的锁
the conditon:
- resouce(locker):
- 互斥:be obtain only one owner
- 不可剥夺: relase by owner
- cycle wait : 循环等待, 持有上一个进程需要的锁, 等待下一个资源释放锁
P0 wait p1.locker , Pn wait P0.locker(n>=1)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
p1:
func getAge:
locker1.lock;
//dosth
locker2.lock
///dosth..
locker2.unlock
locker1.unlock
p2:
func getName():
locker2.lock;
//dosth
locker1.lock ;
//dosth
locker1.unlock;
locker2.unlock;
|
how:
- before: avoid
- doing: react to
- after