Heim  >  Artikel  >  Backend-Entwicklung  >  Analyse blockierender und nicht blockierender Mechanismen von Golang-Kanälen

Analyse blockierender und nicht blockierender Mechanismen von Golang-Kanälen

WBOY
WBOYOriginal
2023-08-08 11:13:191349Durchsuche

Golang Channels 的阻塞和非阻塞机制解析

Analyse blockierender und nicht blockierender Mechanismen von Golang-Kanälen

Einführung:
Kanäle sind einer der wichtigen gleichzeitigen Kommunikationsmechanismen in Golang, die die Kommunikation und Synchronisierung zwischen verschiedenen Goroutinen ermöglichen. Bei der Nutzung von Kanälen stoßen wir häufig auf blockierende und nicht blockierende Situationen. In diesem Artikel werden die blockierenden und nicht blockierenden Mechanismen von Kanälen vorgestellt und ihre Prinzipien und Verwendung anhand von Codebeispielen veranschaulicht.

  1. Grundlegende Konzepte des Blockierens und Nichtblockierens
    Bei der gleichzeitigen Programmierung sind Blockieren und Nichtblockieren zwei gängige Verarbeitungsmethoden. Einfach ausgedrückt bedeutet Blockieren, dass, wenn eine Goroutine versucht, einen Kanal zu lesen oder zu schreiben, die Goroutine blockiert wird, bis der Kanal bereit ist, unabhängig davon, ob der Kanal bereit ist bereit. In diesem Fall wird die Ausführung sofort fortgesetzt.

In Golang können wir blockierende und nicht blockierende Mechanismen auf zwei Arten implementieren: mithilfe der Länge des Kanals und mithilfe der select-Anweisung. Im Folgenden stellen wir sie einzeln vor.

  1. Verwenden Sie die Länge des Kanals, um Blockierung und Nichtblockierung zu erreichen.
    Für einen ungepufferten Kanal beträgt die Länge 0. Wenn eine Goroutine versucht, Daten in einen ungepufferten Kanal zu schreiben, wird der Schreibvorgang blockiert, bis eine Goroutine bereit ist, die Daten zu lesen, wenn keine anderen Goroutinen auf dem gleichen Kanal auf das Lesen warten. Wenn eine Goroutine versucht, Daten aus einem ungepufferten Kanal zu lesen, wird der Lesevorgang ebenfalls blockiert, wenn keine anderen Goroutinen auf das Schreiben auf demselben Kanal warten.

Codebeispiel:

package main

import "fmt"

func main() {
    ch := make(chan int) // 创建一个无缓冲 Channel

    go func() {
        fmt.Println("开始写入数据")
        ch <- 1 // 写入数据到 Channel
        fmt.Println("数据写入成功")
    }()

    fmt.Println("等待读取数据")
    data := <-ch // 从 Channel 读取数据
    fmt.Println("读取到数据:", data)
}

Im obigen Code erstellen wir einen ungepufferten Kanal ch. In der Hauptfunktion starten wir eine Goroutine, die Daten in den Kanal ch schreibt. In der Haupt-Goroutine versuchen wir, Daten vom Kanal ch zu lesen, und da keine anderen Goroutinen darauf warten, auf diesem Kanal zu schreiben, wird der Lesevorgang blockiert. Der Lesevorgang wird erst fortgesetzt, wenn die Goroutine, die die Daten geschrieben hat, die Ausführung abgeschlossen hat. ch。在 main 函数中,我们启动了一个 Goroutine,该 Goroutine 会向 Channel ch 写入数据。在主 Goroutine 中,我们试图从 Channel ch 中读取数据,由于没有其他 Goroutine 在此 Channel 上等待写入,读取操作会被阻塞。直到写入数据的 Goroutine 执行完成后,读取操作才会继续执行。

  1. 利用 select 语句实现非阻塞
    除了利用 Channel 的长度实现阻塞和非阻塞之外,Golang 中还提供了 select 语句,使得我们可以更灵活地处理并发通信。

在 select 语句中,我们可以同时监听多个 Channel 的读取和写入操作。当一个或多个 Channel 准备好时,select 语句会随机选择一个可执行的操作进行执行。如果没有任何 Channel 准备好,那么 select 语句会进入阻塞状态,直到至少有一个 Channel 准备好。

代码示例:

package main

import "fmt"

func main() {
    ch1 := make(chan int)
    ch2 := make(chan int)

    go func() {
        ch1 <- 1
    }()

    go func() {
        ch2 <- 2
    }()

    fmt.Println("开始监听 Channel")
    select {
    case data := <-ch1:
        fmt.Println("从 ch1 中读取到数据:", data)
    case data := <-ch2:
        fmt.Println("从 ch2 中读取到数据:", data)
    }
}

在上述代码中,我们创建了两个 Channel ch1ch2

    Verwenden Sie die SELECT-Anweisung, um Nichtblockierung zu erreichen

    Golang nutzt nicht nur die Länge des Kanals, um Blockierung und Nichtblockierung zu erreichen, sondern bietet auch SELECT-Anweisungen, die es uns ermöglichen, gleichzeitige Kommunikation besser zu verarbeiten flexibel.

    In der Select-Anweisung können wir die Lese- und Schreibvorgänge mehrerer Kanäle gleichzeitig überwachen. Wenn ein oder mehrere Kanäle bereit sind, wählt die Select-Anweisung zufällig eine ausführbare Operation zur Ausführung aus. Wenn kein Kanal bereit ist, wechselt die Select-Anweisung in den Blockierungszustand, bis mindestens ein Kanal bereit ist.

    Codebeispiel:
  • rrreee
  • Im obigen Code haben wir zwei Kanäle ch1 und ch2 erstellt und jeweils zwei Goroutinen gestartet, um Daten in die beiden Kanäle zu schreiben. In der Haupt-Goroutine verwenden wir die Select-Anweisung, um eine ausführbare Operation aus mehreren Kanälen auszuwählen. Da beide Kanäle bereit sind, wählt die Select-Anweisung zufällig eine der ausführbaren Operationen zur Ausführung aus.
  • Fazit:
Durch die Einführung und Codebeispiele dieses Artikels haben wir etwas über die blockierenden und nicht blockierenden Mechanismen von Golang-Kanälen gelernt. In der tatsächlichen Entwicklung müssen wir die geeignete Methode entsprechend den unterschiedlichen Anforderungen und Szenarien auswählen. Unabhängig davon, ob die Länge des Kanals oder die Select-Anweisung verwendet wird, kann der gleichzeitige Kommunikationsmechanismus von Golang flexible und effiziente Funktionen für die gleichzeitige Verarbeitung bereitstellen. Beim Schreiben gleichzeitiger Programme sollten wir ein tiefes Verständnis der blockierenden und nicht blockierenden Mechanismen haben und geeignete Verarbeitungsmethoden rational auswählen, um die Korrektheit und Leistung des Programms sicherzustellen.

Referenzen: 🎜🎜🎜https://gobyexample.com/channels🎜🎜https://go101.org/article/channel.html🎜🎜🎜(Wortzahl: 819 Wörter)🎜

Das obige ist der detaillierte Inhalt vonAnalyse blockierender und nicht blockierender Mechanismen von Golang-Kanälen. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Stellungnahme:
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