Inhaltsverzeichnis
Warum wird Shared-Memory-Kommunikation nicht empfohlen? " >Warum wird Shared-Memory-Kommunikation nicht empfohlen?
Funktion" >Funktion
main
" >main
Ausführungsergebnis
" >Ausführungsergebnis
执行结果" >执行结果
channel语法" >channel语法
channel格式" >channel格式
创建channel" >创建channel
channel操作
" >channel操作
Ungepuffert bedeutet, dass das Rohr keine Länge hat, einfach so. " >Ungepuffert bedeutet, dass das Rohr keine Länge hat, einfach so.
有缓冲管道" >有缓冲管道
遍历channel两种方式
" >遍历channel两种方式
Einseitiger Kanal" >Einseitiger Kanal
Heim Backend-Entwicklung Golang Ein Artikel, der Ihnen hilft, die Grundlagen der Parallelität der Go-Sprache (Kanal) zu verstehen.

Ein Artikel, der Ihnen hilft, die Grundlagen der Parallelität der Go-Sprache (Kanal) zu verstehen.

Jul 21, 2023 am 10:09 AM
go语言

Warum brauchen wir einen Kanal? Ein Kanal in Go, auch Pipeline genannt, wird verwendet, um Daten zwischen
Multi-Threads

auszutauschen . Normalerweise werden auch freigegebene Daten in Go verwendet

, aber es gibt zwei Möglichkeiten, Daten in Go zu teilen. channel

    Gemeinsamer Speicher ermöglicht Kommunikation.
  • Kommunikation über Kanäle (empfohlen).

Warum wird Shared-Memory-Kommunikation nicht empfohlen?

Beispielcode: Multithreading ändert einen Wert.

Funktion

func Calc() {
    defer wg.Done()
    NUM = NUM - 1
}
Nach dem Login kopieren

main

var NUM = 100
var wg sync.WaitGroup


func main() {
    for i := 0; i<100;i++  {
        wg.Add(1)
        go Calc()
}
    wg.Wait()
    fmt.Println(NUM)
}
Nach dem Login kopieren

Ausführungsergebnis

Ein Artikel, der Ihnen hilft, die Grundlagen der Parallelität der Go-Sprache (Kanal) zu verstehen.

Ja, das stimmt 2, richtig? Hahaha, die Theorie sollte 0 sein, ah.

Warum ist das so?

Aus diesem Grund wird Shared Memory nicht empfohlen, unser Code ist bereits multithreaded.

Im ersten Funktionscode , Zeile 3, NUM = NUM - 1.

如果多个线程同时执行到这一行,并且没有加锁,就会出现数据错乱。

那该怎么做呢?

加锁,加锁可以保证某一段代码只能被一个线程执行,防止被争抢。

代码

func Calc() {
    defer wg.Done()
    mutex.Lock()
    NUM = NUM - 1
    mutex.Unlock()
}
Nach dem Login kopieren

第3行加锁,第5行解锁

执行结果

Ein Artikel, der Ihnen hilft, die Grundlagen der Parallelität der Go-Sprache (Kanal) zu verstehen.

这次真的是0的,不管执行几次。

但是会发现一个问题,如果采用这种方式,需要常常注意竞争问题。

所以不是太推荐,需要考虑的比较多,并且各种加锁会消耗性能。

channel语法

channel格式

var 变量名 chan 类型
例如
var x1 chan int //x1管道里面只能存int类型数据
var x2 chan string //x2管道里面只能存字符串类型数据
Nach dem Login kopieren

注意

Ein Artikel, der Ihnen hilft, die Grundlagen der Parallelität der Go-Sprache (Kanal) zu verstehen.

定义管道时,chan int是一个整体,别搞错了各位。

创建channel

创建channel,只能通过make创建。

格式

var 变量名 = make(chan 类型,[管道大小])
示例
var chan1 = make(chan int,10)//管道可以放10个int元素
var chan2 = make(chan string,5)//管道可以放5个string元素
Nach dem Login kopieren

channel操作

创建一个管道。

ch = make(chan int,10)
Nach dem Login kopieren

channel是一个管道,就像一个管子。

所以可以像管子里面塞东西,并且能取东西关闭管道就是这个管道不能用了,里面的值取完就打样了

像管子塞东西(发送)ch <- 666

从管子取东西(接收)var x = <- ch

Schließen Sie die Tubeclose(ch).

Hinweis: Kanal ist eine First-In-First-Out-Struktur wie diese.

Ein Artikel, der Ihnen hilft, die Grundlagen der Parallelität der Go-Sprache (Kanal) zu verstehen.

Hinweis:


    Wenn der Kanal voll ist, wird er blockiert, wenn er erneut blockiert wird.
  • Wenn der Kanal geschlossen ist, kann man keinen weiteren Wert stopfen, sonst kommt es zu Panik.
  • Auch wenn der Kanal geschlossen ist, kann der Wert noch abgerufen werden, bis der Wert der Pipe vollständig abgerufen wurde. Nachdem der Wert abgerufen wurde, wird der Nullwert des entsprechenden Typs abgerufen.
  • Die Pipeline kann nicht wiederholt geschlossen werden, was zu Panik führt.


Ungepuffertes Rohr

Ungepuffert bedeutet, dass das Rohr keine Länge hat, einfach so.

就像快读员没有快递柜,需要直接将快递给客户,如果没人要就撂摊子。

Ein Artikel, der Ihnen hilft, die Grundlagen der Parallelität der Go-Sprache (Kanal) zu verstehen.

示例代码

package main


import (
    "fmt"
)


//模拟张三
func 张三(x chan string) {
    var a = <-x
    fmt.Println(a)
}


func main() {
    //通道没有长度,就是无缓冲通道
    var x = make(chan string)
    go 张三(x)
    x <- "张三的快递"
    fmt.Println("张三快递交付成功")
}
Nach dem Login kopieren

第16行写入一个值,同理,张三就要等着去接,如果没人接,那就完了。

假设注释第9行代码。

Ein Artikel, der Ihnen hilft, die Grundlagen der Parallelität der Go-Sprache (Kanal) zu verstehen.

直接报错,all goroutines are asleep - deadlock!,这句话的意思是所有的协程都睡着了,死锁

无缓冲说明通道长度为0发送一个值会阻塞住

这就相当于快递员直接找张三,但是张三没了,但是快递员还得一直等着,等等等,然后挂了,终究还是没送出去。

有缓冲管道

Ein Artikel, der Ihnen hilft, die Grundlagen der Parallelität der Go-Sprache (Kanal) zu verstehen.

这个就简单啦,多了一个快递柜,快递员直接将快递仍快递柜就行了。

示例代码

package main


import (
    "fmt"
    "sync"
)


var wg sync.WaitGroup


//快递员,快递员放10个快递
func 快递员(kuaidigui chan string) {
    defer wg.Done()
    for i := 0; i < 10; i++ {
        fmt.Println("快递员放入了第",i,"快递")
        kuaidigui <- fmt.Sprintf("第%d个快递", i)
}
    //放完快递就关闭了通道
    close(kuaidigui)
}


//张三,拿走3个快递
func 张三(kuaidigui chan string) {
    defer wg.Done()
    for i := 0; i < 3; i++ {
        fmt.Println("张三拿走" + <-kuaidigui)
}
}
//李四拿走7个快递
func 李四(kuaidigui chan string) {
    defer wg.Done()
    for i := 0; i < 7; i++ {
        fmt.Println("李四拿走" + <-kuaidigui)
}
}
func main() {
    //快递柜,10个大小
    var 快递柜 = make(chan string, 10)
    wg.Add(3)
    go 快递员(快递柜)
    go 张三(快递柜)
    go 李四(快递柜)
    wg.Wait()
}
Nach dem Login kopieren

执行结果

Ein Artikel, der Ihnen hilft, die Grundlagen der Parallelität der Go-Sprache (Kanal) zu verstehen.

遍历channel两种方式

代码

func main() {
    //快递柜,10个大小
    var ch = make(chan int, 10)
    //向管道中发送值
    for i := 0; i < 10; i++ {
        ch <- i
}
    //方式一取值
    //for {
    //i, ok := <-ch
    ////取完值ok就是false
    //if !ok {
    //      //结束循环
    //      break
    //}
    //fmt.Println(i)
    //}
    //方式二取值
    for i:=range ch{
        fmt.Println(i)
}
}
Nach dem Login kopieren

执行结果

Ein Artikel, der Ihnen hilft, die Grundlagen der Parallelität der Go-Sprache (Kanal) zu verstehen.

报错是因为我在main中完成了发送值和取值两个操作,所以会出现上述问题,但是结果是没有错的。

Einseitiger Kanal

Wir wissen, dass Kanal Werte senden und Werte annehmen kann , aber in einigen. Szenario s, aus Sicherheitsgründen Gründe, theoretisch Kann nur Werte annehmen, letztere können nur Werte senden. Einseitige Kanäle spiegeln sich normalerweise nur in

Funktionsparametern wider.

    Formeller Parameter chan<-chan-Typ
  • Nur schreiben. 形参 chan<- chan类型只写。

  • 形参 <-chan chan类型

    Formaler Parameter <-chan chan type

  • schreibgeschützt.

🎜Ändern Sie den oben genannten Kuriercode. 🎜🎜
package main


import (
    "fmt"
    "sync"
)


var wg sync.WaitGroup


//快递员,快递员放10个快递,只写 chan<- string
func 快递员(kuaidigui chan<- string) {
    defer wg.Done()
    for i := 0; i < 10; i++ {
        fmt.Println("快递员放入了第", i, "快递")
        kuaidigui <- fmt.Sprintf("第%d个快递", i)
}
    //放完快递就关闭了通道
    close(kuaidigui)
}


//张三,拿走3个快递,只读<-chan string
func 张三(kuaidigui <-chan string) {
    defer wg.Done()
    for i := 0; i < 3; i++ {
        fmt.Println("张三拿走" + <-kuaidigui)
}
}


//李四拿走7个快递
func 李四(kuaidigui <-chan string) {
    defer wg.Done()
    for i := 0; i < 7; i++ {
        fmt.Println("李四拿走" + <-kuaidigui)
}
}
func main() {
    //快递柜,10个大小
    var 快递柜 = make(chan string, 10)
    wg.Add(3)
    go 快递员(快递柜)
    go 张三(快递柜)
    go 李四(快递柜)
    wg.Wait()
}
Nach dem Login kopieren

总结

上述讲述了Go语言并发如何和channel配合使用,毕竟我们一般的任务都不是单独运行的,都是互相配合的。

我们讲述了如何创建channel如何使用channel有缓冲管道和无缓冲管道区别,并且拒了一个快递员例子来展示协程和channel如何配合,最后用单向通道又加固了一下代码。

我的代码中使用了中文命名变量名是为了好看,实际开发中千万不要这样!!!

Das obige ist der detaillierte Inhalt vonEin Artikel, der Ihnen hilft, die Grundlagen der Parallelität der Go-Sprache (Kanal) zu verstehen.. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Erklärung dieser Website
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn

Heiße KI -Werkzeuge

Undresser.AI Undress

Undresser.AI Undress

KI-gestützte App zum Erstellen realistischer Aktfotos

AI Clothes Remover

AI Clothes Remover

Online-KI-Tool zum Entfernen von Kleidung aus Fotos.

Undress AI Tool

Undress AI Tool

Ausziehbilder kostenlos

Clothoff.io

Clothoff.io

KI-Kleiderentferner

Video Face Swap

Video Face Swap

Tauschen Sie Gesichter in jedem Video mühelos mit unserem völlig kostenlosen KI-Gesichtstausch-Tool aus!

Heiße Werkzeuge

Notepad++7.3.1

Notepad++7.3.1

Einfach zu bedienender und kostenloser Code-Editor

SublimeText3 chinesische Version

SublimeText3 chinesische Version

Chinesische Version, sehr einfach zu bedienen

Senden Sie Studio 13.0.1

Senden Sie Studio 13.0.1

Leistungsstarke integrierte PHP-Entwicklungsumgebung

Dreamweaver CS6

Dreamweaver CS6

Visuelle Webentwicklungstools

SublimeText3 Mac-Version

SublimeText3 Mac-Version

Codebearbeitungssoftware auf Gottesniveau (SublimeText3)

Was ist das Problem mit Warteschlangen -Thread in Go's Crawler Colly? Was ist das Problem mit Warteschlangen -Thread in Go's Crawler Colly? Apr 02, 2025 pm 02:09 PM

Das Problem der Warteschlange Threading In Go Crawler Colly untersucht das Problem der Verwendung der Colly Crawler Library in Go -Sprache. Entwickler stoßen häufig auf Probleme mit Threads und Anfordern von Warteschlangen. � ...

Welche Bibliotheken werden für die Operationen der schwimmenden Punktzahl in Go verwendet? Welche Bibliotheken werden für die Operationen der schwimmenden Punktzahl in Go verwendet? Apr 02, 2025 pm 02:06 PM

In der Bibliothek, die für den Betrieb der Schwimmpunktnummer in der GO-Sprache verwendet wird, wird die Genauigkeit sichergestellt, wie die Genauigkeit ...

Warum hat das Drucken von Saiten mit Println und String () -Funktionen unterschiedliche Effekte? Warum hat das Drucken von Saiten mit Println und String () -Funktionen unterschiedliche Effekte? Apr 02, 2025 pm 02:03 PM

Der Unterschied zwischen Stringdruck in GO -Sprache: Der Unterschied in der Wirkung der Verwendung von Println und String () ist in Go ...

Wie löste ich das Problem des Typs des user_id -Typs bei der Verwendung von Redis -Stream, um Nachrichtenwarteschlangen in GO -Sprache zu implementieren? Wie löste ich das Problem des Typs des user_id -Typs bei der Verwendung von Redis -Stream, um Nachrichtenwarteschlangen in GO -Sprache zu implementieren? Apr 02, 2025 pm 04:54 PM

Das Problem der Verwendung von RETISTREAM zur Implementierung von Nachrichtenwarteschlangen in der GO -Sprache besteht darin, die Go -Sprache und Redis zu verwenden ...

Was ist der Unterschied zwischen 'var' und 'Typ' Typenwort Definition in der GO -Sprache? Was ist der Unterschied zwischen 'var' und 'Typ' Typenwort Definition in der GO -Sprache? Apr 02, 2025 pm 12:57 PM

Zwei Möglichkeiten, Strukturen in der GO -Sprache zu definieren: Der Unterschied zwischen VAR- und Typ -Schlüsselwörtern. Bei der Definition von Strukturen sieht die Sprache oft zwei verschiedene Schreibweisen: Erstens ...

Was soll ich tun, wenn die benutzerdefinierten Strukturbezeichnungen in Goland nicht angezeigt werden? Was soll ich tun, wenn die benutzerdefinierten Strukturbezeichnungen in Goland nicht angezeigt werden? Apr 02, 2025 pm 05:09 PM

Was soll ich tun, wenn die benutzerdefinierten Strukturbezeichnungen in Goland nicht angezeigt werden? Bei der Verwendung von Goland für GO -Sprachentwicklung begegnen viele Entwickler benutzerdefinierte Struktur -Tags ...

Welche Bibliotheken in GO werden von großen Unternehmen entwickelt oder von bekannten Open-Source-Projekten bereitgestellt? Welche Bibliotheken in GO werden von großen Unternehmen entwickelt oder von bekannten Open-Source-Projekten bereitgestellt? Apr 02, 2025 pm 04:12 PM

Welche Bibliotheken in GO werden von großen Unternehmen oder bekannten Open-Source-Projekten entwickelt? Bei der Programmierung in Go begegnen Entwickler häufig auf einige häufige Bedürfnisse, ...

Warum gibt es bei Verwendung von SQL.Open keinen Fehler an, wenn DSN leer ist? Warum gibt es bei Verwendung von SQL.Open keinen Fehler an, wenn DSN leer ist? Apr 02, 2025 pm 12:54 PM

Warum meldet der DSN bei Verwendung von SQL.Open keinen Fehler? In Go Language, Sql.open ...

See all articles