golang 锁

蛐蛐儿阳 · · 1892 次点击 · · 开始浏览    
这是一个创建于 的文章,其中的信息可能已经有所发展或是发生改变。

手册中有点难懂,自己写个例子。

package main

import (
    "fmt"
    "time"
)

func main() {
    // 不想用time sleep
    // 挂起通道 不让主main退出,我想看结果
    //ch := make(chan bool)

    public := 1


    go func() {
        fmt.Println("执行了1操作")
        for i := 0; i < 2; i++ {
            time.Sleep(time.Microsecond * 100)
            public++
            fmt.Println(public)
        }
    }()

    go func() {
        fmt.Println("执行了2操作")
        for i := 0; i < 2; i++ {
            time.Sleep(time.Microsecond * 100)
            public--
            fmt.Println(public)
        }
    }()

    time.Sleep(time.Second * 2)
    println(public)
}

写完后发现没手册中的好理解, 输出

执行了2操作
执行了1操作
2
1
2
1
1

反正不管怎么执行,最后结果1不变, 但四次运算,输出顺序是乱的。
我们加锁。

package main

import (
    "fmt"
    "sync"
    "time"
)

func main() {
    // 不想用time sleep
    // 挂起通道 不让主main退出,我想看结果
    //ch := make(chan bool)

    // 记住声明方式
    var lock sync.Mutex


    public := 1


    go func() {
        lock.Lock()
        fmt.Println("执行了1操作")
        for i := 0; i < 2; i++ {
            time.Sleep(time.Microsecond * 100)
            public++
            fmt.Println(public)
        }
        lock.Unlock()
    }()

    go func() {
        lock.Lock()
        fmt.Println("执行了2操作")
        for i := 0; i < 2; i++ {
            time.Sleep(time.Microsecond * 100)
            public--
            fmt.Println(public)
        }
        lock.Unlock()
    }()

    time.Sleep(time.Second * 2)
    println(public)
}

注意这里锁是加了,但并不保证go1和go2哪个先开始。
但,结果就好预测了, go1先开始,输出

执行了1操作
2
3
执行了2操作
2
1
1

go2先开始,输出

执行了2操作
0
-1
执行了1操作
0
1
1

就只有这两种可能。
这里又学到一点知识。在主函数里添加

public = public * 2

他总是先*2,再执行,当然要放到sleep前面。


有疑问加站长微信联系(非本文作者)

本文来自:简书

感谢作者:蛐蛐儿阳

查看原文:golang 锁

入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889

1892 次点击  
加入收藏 微博
暂无回复
添加一条新回复 (您需要 登录 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传