亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Go 并發(fā)編程Goroutine的實現(xiàn)示例

 更新時間:2024年12月16日 11:16:27   作者:比豬聰明  
Go語言中的并發(fā)編程主要通過Goroutine和Channel來實現(xiàn),本文就來介紹一下Go 并發(fā)編程的實現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

進程(Process),線程(Thread),協(xié)程(Goroutine,也叫輕量級線程)

進程

進程是一個程序在一個數(shù)據(jù)集中的一次動態(tài)執(zhí)行過程,進程一般由程序,數(shù)據(jù)集,進程控制塊三部分組成

線程

線程也叫輕量級進程,他是一個基本的CPU執(zhí)行單元,也就是程序執(zhí)行過程中的最小單元,由線程ID,程序計數(shù)器,寄存器集合和堆棧共同組成的,一個進程可以包含多個線程

協(xié)程

協(xié)程是一種用戶態(tài)的輕量級線程,又稱微線程,協(xié)程的調(diào)度完全由用戶控制

一、主Goroutine

封裝main函數(shù)的Goroutine被稱為主Goroutine

主Goroutine所做的事情并不是執(zhí)行main函數(shù)那么簡單,它首先要做的是設(shè)定每一個goroutine所能申請的??臻g的最大尺寸,在32位計算機系統(tǒng)中此最大尺寸為250MB,而在64位計算機系統(tǒng)中此尺寸為1GB,如果有某個Goroutine的??臻g尺寸大于這個限制,那么運行時系統(tǒng)就會引發(fā)一個棧溢出(stack overflow)的運行時恐慌,隨后這個go程序的運行也會終止

此后主Goroutine會進行一系列的初始化工作:

1、創(chuàng)建一個特殊的defer語句,用于在主Goroutine退出時做必要的善后處理,因為主Goroutine也可能非正常結(jié)束

2、啟動專用于在后臺清掃內(nèi)存垃圾的Goroutine,并設(shè)置GC可用的表示

3、執(zhí)行main包中所引用包的init函數(shù)

4、執(zhí)行main函數(shù)

二、Goroutine

GO中使用Goroutine來實現(xiàn)并發(fā)

Goroutine是與其他函數(shù)或方法同時運行的函數(shù)或方法,與線程相比創(chuàng)建goroutine的成本很小,他就是一段代碼,一個函數(shù)入口,以及在堆上為其分配一個堆棧(初始大小為4k,會隨著程序的執(zhí)行自動增長刪除)。

在GO語言中使用goroutine,在調(diào)用函數(shù)或者方法前面加上go關(guān)鍵字即可  

package main

import "fmt"

func main() {

	// 使用go關(guān)鍵字使用goroutine調(diào)用hello函數(shù)
	go hello()
	for i := 0; i < 150000; i++ {
		//fmt.Println("main-", i)
	}
}

func hello() {
	for i := 0; i < 10; i++ {
		fmt.Println("hello-----------", i)
	}
}

/*
    此處代碼可設(shè)置main協(xié)程for循環(huán)次數(shù)的大小觀測go協(xié)程調(diào)用hello情況
*/
  • 當(dāng)新的Goroutine開始時,Goroutine調(diào)用立即返回,與函數(shù)不同,go不等待Goroutine執(zhí)行結(jié)束

  • 當(dāng)Goroutine調(diào)用,并且Goroutine的任何返回值被忽略之后,go立即執(zhí)行到下一行代碼

  • mian的Goroutine應(yīng)該為其他的Goroutine執(zhí)行,如果main的Goroutine終止了,程序?qū)⒈唤K止,而其他的Goroutine將不會運行

三、runtime

  • 獲取系統(tǒng)信息

  • schedule調(diào)度讓出時間片,讓別的goroutine先執(zhí)行

  • Goexit   //終止當(dāng)前的goroutine 

Go 語言的 runtime 包提供了與 Go 運行時環(huán)境交互的各種功能。這個包允許你控制和檢查程序的運行時行為,包括但不限于:

  • 垃圾回收(Garbage Collection):可以手動觸發(fā)垃圾回收,或者調(diào)整垃圾回收的策略。

  • 并發(fā)控制:提供了包括 Gosched() 在內(nèi)的方法來控制 goroutine 的調(diào)度。

  • 程序退出:可以正常或非正常地退出程序。

  • 堆棧管理:可以獲取當(dāng)前 goroutine 的堆棧信息。

  • 環(huán)境變量:讀取和設(shè)置環(huán)境變量。

  • 系統(tǒng)信號:處理操作系統(tǒng)信號。

  • CPU 信息:獲取 CPU 的數(shù)量和相關(guān)信息。

  • 內(nèi)存分配:可以手動分配和釋放內(nèi)存。

  • 性能監(jiān)控:可以監(jiān)控程序的 CPU 使用情況。

以下是一些 runtime 包中常用函數(shù)的簡要說明:

  • runtime.GOMAXPROCS:設(shè)置最大可運行的操作系統(tǒng)線程數(shù)。
  • runtime.NumCPU:返回機器的 CPU 核心數(shù)。
  • runtime.NumGoroutine:返回當(dāng)前運行的 goroutine 數(shù)量。
  • runtime.Gosched:讓出 CPU 時間片,使得其他 goroutine 可以運行。
  • runtime.Goexit:退出當(dāng)前的 goroutine。
  • runtime.KeepAlive:確保某個 goroutine 不會被垃圾回收。
  • runtime.SetFinalizer:為對象設(shè)置終結(jié)器,當(dāng)垃圾回收器準(zhǔn)備回收該對象時,會調(diào)用該終結(jié)器。
  • runtime.GC:強制運行垃圾回收器。 
package main

import (
	"fmt"
	"runtime"
)

func main() {

	//獲取系統(tǒng)信息
	fmt.Println("獲取GOROOT目錄", runtime.GOROOT())
	fmt.Println("獲取操作系統(tǒng)", runtime.GOOS)
	fmt.Println("獲取CPU", runtime.NumCPU())

	//Goroutine 調(diào)度
	go func() {
		for i := 0; i < 100; i++ {
			fmt.Println("Goroutine---", i)
		}
	}()

	for i := 0; i < 100; i++ {
		//讓出時間片,讓別的Goroutine先執(zhí)行,不一定可以讓成功
		runtime.Gosched()
		fmt.Println("main---", i)
	}
}

runtime.Gosched() 是 Go 語言運行時庫中的一個函數(shù),它用于讓出 CPU 時間片,讓其他 goroutine(輕量級線程)有機會執(zhí)行。這通常用于避免阻塞或減少阻塞的持續(xù)時間,尤其是在長時間運行的 goroutine 中,你可能會在適當(dāng)?shù)牡胤秸{(diào)用 Gosched 來讓出 CPU,以避免長時間占用 CPU 導(dǎo)致其他 goroutine 饑餓。

以下是 runtime.Gosched() 函數(shù)的一些使用場景:

  • 避免饑餓:在長時間運行的循環(huán)中,如果確定當(dāng)前 goroutine 可能不會被阻塞,可以調(diào)用 Gosched 來讓出 CPU。

  • 控制執(zhí)行順序:在某些情況下,你可能希望控制 goroutine 的執(zhí)行順序,通過 Gosched 可以給其他 goroutine 運行的機會。

  • 減少 CPU 使用:在某些 I/O 密集型操作中,如果當(dāng)前 goroutine 主要是等待 I/O 操作完成,調(diào)用 Gosched 可以讓出 CPU,減少不必要的 CPU 使用。

  • 避免死鎖:在某些復(fù)雜的 goroutine 調(diào)度中,如果擔(dān)心死鎖問題,可以在適當(dāng)?shù)牡胤秸{(diào)用 Gosched 來減少死鎖的風(fēng)險。 

package main

import (
	"fmt"
	"runtime"
	"time"
)

func main() {
	/*
		因為goroutine2延時了一定時間,如果goroutine1不讓出CPU時間片那么必先執(zhí)行完成
	*/
	go func() {
		for i := 0; i < 5; i++ {
			runtime.Gosched() // 讓出 CPU 時間片
			fmt.Println("Goroutine 1:", i)
		}
	}()

	go func() {
		for i := 0; i < 5; i++ {
			time.Sleep(100 * time.Millisecond)
			fmt.Println("Goroutine 2:", i)
		}
	}()

	time.Sleep(3 * time.Second) // 等待兩個 goroutine 執(zhí)行完畢
}

請注意,過度使用 Gosched 可能會導(dǎo)致性能下降,因為頻繁的調(diào)度會消耗額外的 CPU 資源。因此,應(yīng)該在仔細(xì)考慮后,根據(jù)實際需要來使用 Gosched。

查看協(xié)程數(shù)&CUP數(shù)

package main

import (
	"fmt"
	"runtime"
	"sync"
)

func main() {
	var wg sync.WaitGroup //創(chuàng)建并發(fā)組
	fmt.Printf("當(dāng)前運行的goroutine數(shù)量: %d\n", runtime.NumGoroutine())
	//創(chuàng)建10個協(xié)程
	for i := 0; i < 10; i++ {
		wg.Add(1)
		go func(id int) {
			defer wg.Done()
			fmt.Printf("第 %d 個協(xié)程在 running\n", id)
		}(i)
	}
	fmt.Printf("當(dāng)前運行的goroutine數(shù)量: %d\n", runtime.NumGoroutine())
	wg.Wait()

	fmt.Printf("CPU核心數(shù): %d\n", runtime.NumCPU())
}

四、互斥鎖 

在并發(fā)編程中會遇到的臨界資源安全問題,可以采用互斥鎖的方式來解決,后面也可通過通過通道channel來解決

臨界資源:指并發(fā)環(huán)境中多個進程、線程、協(xié)程共享的資源

使用sync包下的鎖解決臨界資源安全問題(Mutex) 

package main

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

// 定義全局變量 票庫存為10張
var tickets int = 10

// 創(chuàng)建鎖
var mutexs sync.Mutex

func main() {
	//三個窗口同時售票
	go saleTicket("售票口1")
	go saleTicket("售票口2")
	go saleTicket("售票口3")
	time.Sleep(time.Second * 15) //等待售票完

}

// 售票函數(shù)
func saleTicket(name string) {
	for {
		// 在檢查之前上鎖
		mutexs.Lock()
		if tickets > 0 {
			time.Sleep(time.Second)
			fmt.Printf("%s剩余的票數(shù)為:%d\n", name, tickets)
			tickets--
		} else {
			fmt.Println("票已售完")
			break
		}
		//操作結(jié)束后解鎖
		mutexs.Unlock()
	}
}

sync包下的同步等待組(WaitGroup)

package main

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

var w sync.WaitGroup

func main() {
	// 公司最后關(guān)門的人   0
	// wg.Add(2) 判斷還有幾個線程、計數(shù)  num=2
	// wg.Done() 我告知我已經(jīng)結(jié)束了  -1
	w.Add(2)

	go test11()
	go test22()

	fmt.Println("main等待ing")
	w.Wait() // 等待 wg 歸零,才會繼續(xù)向下執(zhí)行
	fmt.Println("end")

	// 理想狀態(tài):所有協(xié)程執(zhí)行完畢之后,自動停止。
	//time.Sleep(3 * time.Second)

}
func test11() {
	for i := 0; i < 5; i++ {
		time.Sleep(1 * time.Second)
		fmt.Println("test1--", i)
	}
	w.Done()
}
func test22() {
	defer w.Done()
	for i := 0; i < 5; i++ {
		fmt.Println("test2--", i)
	}
}

五、Channel通道 

 不要以共享內(nèi)存的方式通信,而要以通信的方式共享內(nèi)存

 通道可以被認(rèn)為是Goroutines通信的管道,類似于管道中的水從一端到另一端的流動,數(shù)據(jù)可以從一端發(fā)送到另一端,通過通道接收,GO語言中建議使用Channel通道來實現(xiàn)Goroutines之間的通信

GO從語言層面保證同一個時間只有一個goroutine能夠訪問channel里面的數(shù)據(jù),使用channel來通信,通過通信來傳遞內(nèi)存數(shù)據(jù),使得內(nèi)存數(shù)據(jù)在不同的goroutine中傳遞,而不是使用共享內(nèi)存來通信

每個通道都有與其相關(guān)的類型,類型是通道允許傳輸?shù)臄?shù)據(jù)類型(通道的零值為nil,nil通道沒有任何用處,因此通道必須使用類似于map和切片的方法定義)

一個通道發(fā)送和接收數(shù)據(jù)默認(rèn)是阻塞的,當(dāng)一個數(shù)據(jù)被發(fā)送到通道時,在發(fā)送語句中被阻塞,直到另一個Goroutine從通道中讀取數(shù)據(jù) 

關(guān)閉通道

發(fā)送者可以通過關(guān)閉通道來通知接收方不會有更多的數(shù)據(jù)被發(fā)送到通道

close(ch)

 接收者可以在接收來自通道的數(shù)據(jù)時使用額外的變量來檢查通道是否已關(guān)閉

v,ok := <- ch

當(dāng)ok的值為true,表示成功的從通道中讀取了一個數(shù)據(jù)value,通道關(guān)閉時仍然可以讀(存)數(shù)據(jù)當(dāng)ok的值為false,表示從一個封閉的通道讀取數(shù)據(jù),從閉通道讀取的數(shù)據(jù)將是通道類型的零值 

緩沖通道

緩沖通道是指一個通道,帶有一個緩沖區(qū),發(fā)送到一個緩沖通道只有在緩沖區(qū)滿時才被阻塞,類似的,從緩沖通道接收的信息只有在為空時才會被阻塞,可以通過將額外的容量參數(shù)傳遞給make函數(shù)來創(chuàng)建緩沖通道,該函數(shù)指定緩沖區(qū)的大小

package main

import (
	"fmt"
	"strconv"
	"time"
)

func main() {
	//定義通道可以寫10個數(shù)據(jù)
	ch := make(chan string, 10)
	go test3(ch)

	for v := range ch {
		fmt.Println(v)
	}
}

func test3(ch chan string) {
	for i := 0; i < 5; i++ {
		time.Sleep(time.Second)
		fmt.Println("通道內(nèi)寫入數(shù)據(jù)", "tset--"+strconv.Itoa(i))
		ch <- "tset--" + strconv.Itoa(i)
	}
	close(ch)//如果不關(guān)閉協(xié)程,主協(xié)程的for循環(huán)一值阻塞,知道報錯“fatal error: all goroutines are asleep - deadlock!”
}

定向通道

單向通道也就是定向通道,這些通道只能發(fā)送數(shù)據(jù)或者接收數(shù)據(jù) 

package main

import (
	"fmt"
	"time"
)

func main() {
	ch := make(chan int)
	go writerOnly(ch)
	go readOnly(ch)

	time.Sleep(time.Second * 2)
}

// 只讀,指只允許管道讀入/寫出數(shù)據(jù)
func writerOnly(ch chan<- int) {
	ch <- 10
}

// 只寫,指指只允許管道讀出/寫入數(shù)據(jù)
func readOnly(ch <-chan int) {
	temp := <-ch
	fmt.Println(temp)
}

Select

  • 每個case都必須是一個通道的操作

  • 如果任意某個通信可以進行,他就執(zhí)行,其他被忽略

  • 如果有多個case都可以運行,Select會隨機公平的選出一個執(zhí)行

否則

  • 如果有default子句,則執(zhí)行該語句

  • 如果沒有default字句,select將阻塞,直到某個通信可以運行,GO不會重新對channel或值進行求值

package main

import (
	"fmt"
	"time"
)

func main() {

	ch1 := make(chan int)
	ch2 := make(chan int)

	go func() {
		time.Sleep(time.Second * 2)
		ch1 <- 100
	}()

	go func() {
		time.Sleep(time.Second * 2)
		ch2 <- 200
	}()

	select {
	case num1 := <-ch1:
		fmt.Println("ch1--", num1)
	case num2 := <-ch2:
		fmt.Println("ch2--", num2)
	}
    //沒有default,則等待通道(阻塞),因為select{}本身是阻塞的
}

 利用通道解決臨界資源安全問題

package main

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

// 定義全局chan,存儲票總數(shù)
var totalTickets chan int
var wg sync.WaitGroup

func main() {

	// 初始化票數(shù)量:總票數(shù)10張
	totalTickets = make(chan int, 2)
	totalTickets <- 10

	wg.Add(3)

	go sell("售票口1")
	go sell("售票口2")
	go sell("售票口3")

	wg.Wait()

	fmt.Println("買完了,下班")

}

func sell(name string) {
	defer wg.Done()

	for { //for循環(huán)表示一直在賣,一直在營業(yè)
		residue, ok := <-totalTickets
		if !ok {
			fmt.Printf("%s: 關(guān)閉\n", name)
			break
		}
		if residue > 0 {
			time.Sleep(time.Second * 1)
			totalTickets <- residue - 1 //
			fmt.Println(name, "售出1張票,余票:", residue)
		} else {
			//進入此處時票已經(jīng)買完了,因為for循環(huán)一進來售票窗口就檢查是否還有票,假如最后賣完的是售票口3,那么銷售窗口1跟2就會判斷還有沒有
			fmt.Printf("%s: 關(guān)閉\n", name)
			close(totalTickets)
			break
		}
	}
}

到此這篇關(guān)于Go 并發(fā)編程Goroutine的實現(xiàn)示例的文章就介紹到這了,更多相關(guān)Go 并發(fā)Goroutine內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家! 

相關(guān)文章

  • 一步步教你在Linux上安裝Go語言環(huán)境

    一步步教你在Linux上安裝Go語言環(huán)境

    本文將介紹如何在Linux操作系統(tǒng)下搭建Go語言環(huán)境,Go語言是一種開源的編程語言,具有高效、簡潔和并發(fā)性強的特點,適用于開發(fā)各種類型的應(yīng)用程序,搭建Go語言環(huán)境是開始學(xué)習(xí)和開發(fā)Go語言項目的第一步,本文將詳細(xì)介紹安裝Go語言、配置環(huán)境變量以及驗證安裝是否成功的步驟
    2023-10-10
  • Golang Goroutine的使用

    Golang Goroutine的使用

    這篇文章主要介紹了Golang Goroutine的使用,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-10-10
  • Golang中優(yōu)秀的消息隊列NSQ基礎(chǔ)安裝及使用詳解

    Golang中優(yōu)秀的消息隊列NSQ基礎(chǔ)安裝及使用詳解

    這篇文章主要介紹了Golang中優(yōu)秀的消息隊列NSQ基礎(chǔ)安裝及使用詳解,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-12-12
  • Go語言實現(xiàn)websocket推送程序

    Go語言實現(xiàn)websocket推送程序

    這篇文章主要介紹了Go語言實現(xiàn)websocket推送程序,WebSocket是基于TCP的一個雙向傳輸數(shù)據(jù)的協(xié)議,和HTTP協(xié)議一樣,是在應(yīng)用層的,他的出現(xiàn),是為了解決網(wǎng)頁進行持久雙向傳輸數(shù)據(jù)的問題
    2023-01-01
  • GoLang中Strconv庫有哪些常用方法

    GoLang中Strconv庫有哪些常用方法

    這篇文章主要介紹了GoLang中Strconv庫有哪些常用方法,strconv庫實現(xiàn)了基本數(shù)據(jù)類型與其字符串表示的轉(zhuǎn)換,主要有以下常用函數(shù):?Atoi()、Itia()、parse系列、format系列、append系列
    2023-01-01
  • Golang通脈之流程控制詳情

    Golang通脈之流程控制詳情

    這篇文章主要介紹了Golang通脈之流程控制,流程控制是每種編程語言控制邏輯走向和執(zhí)行次序的重要部分,Go語言中最常用的流程控制有if和for,而switch和goto主要是為了簡化代碼,下面文章將詳細(xì)介紹改該內(nèi)容,需要的朋友可以參考一下
    2021-10-10
  • Golang依賴注入工具digo的使用詳解

    Golang依賴注入工具digo的使用詳解

    這篇文章主要為大家詳細(xì)介紹了Golang中依賴注入工具digo的使用,文中的示例代碼講解詳細(xì),具有一定的學(xué)習(xí)價值,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下
    2023-06-06
  • golang?gorm的Callbacks事務(wù)回滾對象操作示例

    golang?gorm的Callbacks事務(wù)回滾對象操作示例

    這篇文章主要為大家介紹了golang?gorm的Callbacks事務(wù)回滾對象操作示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步早日升職加薪
    2022-04-04
  • 淺談Go語言中字符串和數(shù)組

    淺談Go語言中字符串和數(shù)組

    這篇文章主要簡單介紹了Go語言中字符串和數(shù)組的使用方法和申明方式,需要的朋友可以參考下
    2015-01-01
  • Golang 之協(xié)程的用法講解

    Golang 之協(xié)程的用法講解

    這篇文章主要介紹了Golang 之協(xié)程的用法講解,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-04-04

最新評論