你可能不知道的關于golang 的10件事情

英文原文

匿名結構體

最常見的匿名用法,不用單獨定義一個結構體類型

    var config struct {
        APIKey      string
        OAuthConfig oauth.Config
    }

    config.APIKey = "BADC0C0A"

匿名結構體定義時并初始化

    data := struct {
        Title string
        Users []*User
    }{
        title,
        users,
    }
    err := tmpl.Execute(w, data)

匿名結構體slice定義并初始化

    var indexRuneTests = []struct {
        s    string
        rune rune
        out  int
    }{
        {"a A x", 'A', 2},
        {"some_text=some_value", '=', 9},
        {"?a", 'a', 3},
        {"a??b", '?', 4},
    }

嵌套互斥鎖

    var hits struct {
        sync.Mutex
        n int
    }

    hits.Lock()
    hits.n++
    hits.Unlock()

嵌套結構體

    type Item struct {
        Title string
        URL   string
    }

    type Response struct {
        Data struct {
            Children []struct {
                Data Item
            }
        }
    }

命令行 go doc

在命令行中,可以通過go doc 查看包相關接口信息

wdy@wdy:~/learn/program-learn/golang$ go doc sync
package sync // import "sync"

Package sync provides basic synchronization primitives such as mutual
exclusion locks. Other than the Once and WaitGroup types, most are intended
for use by low-level library routines. Higher-level synchronization is
better done via channels and communication.

Values containing the types defined in this package should not be copied.

func NewCond(l Locker) *Cond
type Cond struct { ... }
type Locker interface { ... }
type Mutex struct { ... }
type Once struct { ... }
type Pool struct { ... }
type RWMutex struct { ... }
type WaitGroup struct { ... }
wdy@wdy:~/learn/program-learn/golang$ go doc sync Mutex
type Mutex struct {
        // Has unexported fields.
}
    A Mutex is a mutual exclusion lock. Mutexes can be created as part of other
    structures; the zero value for a Mutex is an unlocked mutex.


func (m *Mutex) Lock()
func (m *Mutex) Unlock()
wdy@wdy:~/learn/program-learn/golang$ go doc sync.mutex.lock
func (m *Mutex) Lock()
    Lock locks m. If the lock is already in use, the calling goroutine blocks
    until the mutex is available.

另外還有一個厲害的工具 ,執(zhí)行下面命令

godoc -http :6061

會打開一個網(wǎng)頁如下:

2017-07-11 19:16:11屏幕截圖.png

可以方便的查詢各個package的相關知識。

在同一個channel上進行讀寫操作

battle是一個阻塞channel,當多個goroutine執(zhí)行warrior的時候,最先執(zhí)行到select的goroutine進入到

case battle <- name:

此時其他goroutine會阻塞在select上,name進入battle的瞬間,阻塞等待的goroutine的其中一個會進入到

case opponent := <-battle:

battle數(shù)據(jù)被讀取后,阻塞等待的剩余的goroutine的其中一個會進入到

case battle <- name:

按照如上流程反復直到所有goroutine執(zhí)行完畢。

var battle = make(chan string)

func warrior(name string, done chan struct{}) {
    select {
    case opponent := <-battle:
        fmt.Printf("%s beat %s\n", name, opponent)
    case battle <- name:
        // I lost :-(
    }
    done <- struct{}{}
}

func main() {
    done := make(chan struct{})
    langs := []string{"Go", "C", "C++", "Java", "Perl", "Python"}
    for _, l := range langs {
        go warrior(l, done)
    }
    for _ = range langs {
        <-done
    }
}

可以看到每次結果都是不一樣的

wdy@wdy:~/learn/program-learn/golang$ go run learnchannel.go
Go beat Python
C++ beat C
Java beat Perl
wdy@wdy:~/learn/program-learn/golang$ go run learnchannel.go
Perl beat Java
Python beat Go
C beat C++
wdy@wdy:~/learn/program-learn/golang$ go run learnchannel.go
C++ beat C
Java beat Perl
Python beat Go
wdy@wdy:~/learn/program-learn/golang$ go run learnchannel.go
Go beat Python
Perl beat Java
C beat C++
wdy@wdy:~/learn/program-learn/golang$ go run learnchannel.go
Python beat Go
C beat C++
Java beat Perl
wdy@wdy:~/learn/program-learn/golang$ go run learnchannel.go
Go beat Python
Perl beat Java
C beat C++
wdy@wdy:~/learn/program-learn/golang$ go run learnchannel.go
Go beat Python
C beat C++
Java beat Perl

利用close來向其他goroutine廣播

func waiter(i int, block, done chan struct{}) {
    time.Sleep(time.Duration(rand.Intn(3000)) * time.Millisecond)
    fmt.Println(i, "waiting...")
    <-block
    fmt.Println(i, "done!")
    done <- struct{}{}
}

func main() {
    block, done := make(chan struct{}), make(chan struct{})
    for i := 0; i < 4; i++ {
        go waiter(i, block, done)
    }
    time.Sleep(5 * time.Second)
    close(block)
    for i := 0; i < 4; i++ {
        <-done
    }
}

結果

2 waiting...
1 waiting...
3 waiting...
0 waiting...
0 done!
3 done!
2 done!
1 done!

利用nil通道的特性

goroutine對于值為nil的channel進行接受或發(fā)送操作時,會永久block

type Work struct {
    Job string
}

func (w Work) Do() {
    fmt.Println("do", w.Job)
}

func (w Work) Refuse() {
    fmt.Println(w.Job + "stopped")
}

func makeWork(ch chan Work) {
    for {
        time.Sleep(500 * time.Millisecond)
        ch <- Work{Job: "job"}
    }
}

func worker(i int, ch chan Work, quit chan struct{}) {
    for {
        select {
        case w := <-ch:
            if quit == nil {
                w.Refuse()
                fmt.Println("worker", i, "refused", w)
                break
            }
            w.Do()
            fmt.Println("worker", i, "processed", w)
        case <-quit:
            fmt.Println("worker", i, "quitting")
            quit = nil
        }
    }
}

結果

do job
worker 0 processed {job}
do job
worker 1 processed {job}
do job
worker 2 processed {job}
do job
worker 3 processed {job}
do job
worker 0 processed {job}
do job
worker 1 processed {job}
do job
worker 2 processed {job}
do job
worker 3 processed {job}
do job
worker 0 processed {job}
worker 0 quitting
worker 1 quitting
worker 2 quitting
worker 3 quitting
jobstopped
worker 0 refused {job}
jobstopped
worker 1 refused {job}
jobstopped
worker 2 refused {job}
jobstopped
worker 3 refused {job}

后記

之前一直在CSDN上寫文章,后面會逐步轉換到簡書上,還請大家多多支持。

最后編輯于
?著作權歸作者所有,轉載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

相關閱讀更多精彩內(nèi)容

  • Goroutine是Go里的一種輕量級線程——協(xié)程。相對線程,協(xié)程的優(yōu)勢就在于它非常輕量級,進行上下文切換的代價非...
    witchiman閱讀 5,153評論 0 9
  • 控制并發(fā)有三種種經(jīng)典的方式,一種是通過channel通知實現(xiàn)并發(fā)控制 一種是WaitGroup,另外一種就是Con...
    wiseAaron閱讀 10,831評論 4 34
  • 能力模型 選擇題 [primary] 下面屬于關鍵字的是()A. funcB. defC. structD. cl...
    _張曉龍_閱讀 25,142評論 14 224
  • 本文翻譯自Sameer Ajmani的文章《Go Concurrency Patterns: Pipelines ...
    大蟒傳奇閱讀 3,981評論 0 15
  • Go語言做Web編程非常方便,并且在開發(fā)效率和程序運行效率方面都非常優(yōu)秀。相比于Java,其最大的優(yōu)勢就是簡便易用...
    暗黑破壞球嘿哈閱讀 9,177評論 6 66

友情鏈接更多精彩內(nèi)容