日本免费高清视频-国产福利视频导航-黄色在线播放国产-天天操天天操天天操天天操|www.shdianci.com

學(xué)無先后,達者為師

網(wǎng)站首頁 編程語言 正文

GO語言協(xié)程創(chuàng)建使用并通過channel解決資源競爭_Golang

作者:Jeff的技術(shù)棧 ? 更新時間: 2022-06-14 編程語言

創(chuàng)建協(xié)程

goroutine是go的設(shè)計核心,就是協(xié)程
主協(xié)程終止了,子協(xié)程也終止

package main
import (
	"fmt"
	"time"
)
func newTask() {
	for {
		fmt.Println("this is a newTask")
		time.Sleep(time.Second) //延時1s
	}
}
func main() {
	go newTask() //新建一個協(xié)程, 新建一個任務(wù)
	for {
		fmt.Println("this is a main goroutine")
		time.Sleep(time.Second) //延時1s
	}
}

主協(xié)程終止,子協(xié)程也終止

package main
import (
	"fmt"
	"time"
)
//主協(xié)程退出了,其它子協(xié)程也要跟著退出
func main() {
	go func() {
		i := 0
		for {
			i++
			fmt.Println("子協(xié)程 i = ", i)
			time.Sleep(time.Second)
		}
	}() //別忘了()
	i := 0
	for {
		i++
		fmt.Println("main i = ", i)
		time.Sleep(time.Second)
		if i == 2 {
			break
		}
	}
}

runtime包

Gosched讓出CPU時間片

等待其他協(xié)程執(zhí)行完

runtime.Gosched()用于讓出CPU時間片,讓出當(dāng)前goroutine(協(xié)程)的執(zhí)行權(quán)限,調(diào)度器安排其他等待的任務(wù)運行,并在下次某個時候從該位置恢復(fù)執(zhí)行。
類似:接力賽,A跑了一會碰到代碼runtime.Gosched()就把接力棒交給B,A歇著,B繼續(xù)跑

案例:

package main
import (
	"fmt"
	"runtime"
)
func main() {
	go func() {
		for i := 0; i < 5; i++ {
			fmt.Println("go")
		}
	}()
	for i := 0; i < 2; i++ {
		//讓出時間片,先讓別的協(xié)議執(zhí)行,它執(zhí)行完,再回來執(zhí)行此協(xié)程
		runtime.Gosched() 
		fmt.Println("hello")
	}
}


go
go
go
go
go
hello
hello

Goexit立即結(jié)束當(dāng)前協(xié)程

runtime.Goexit()  //立即結(jié)束當(dāng)前協(xié)程

案例:

package main
import (
	"fmt"
	"runtime"
)
func test() {
	defer fmt.Println("ccccccccccccc")
	//return //終止此函數(shù)
	runtime.Goexit() //終止所在的協(xié)程
	fmt.Println("dddddddddddddddddddddd")
}
func main() {
	//創(chuàng)建新建的協(xié)程
	go func() {
		fmt.Println("aaaaaaaaaaaaaaaaaa")

		//調(diào)用了別的函數(shù)
		test()
		fmt.Println("bbbbbbbbbbbbbbbbbbb")
	}() //別忘了()
	//特地寫一個死循環(huán),目的不讓主協(xié)程結(jié)束
	for {
	}
}
aaaaaaaaaaaaaaaaaa
ccccccccccccc

GOMAXPROCS設(shè)置并行CPU核數(shù)最大值,并返回之前的值

runtime.GOMAXPROCS() //設(shè)置并行CPU核數(shù)最大值,并返回之前的值
package main
import (
	"fmt"
	"runtime"
)
func main() {
	//n := runtime.GOMAXPROCS(1) //指定以1核運算
	n := runtime.GOMAXPROCS(2) //指定以8核運算
	fmt.Println("n = ", n)
	for {
		go fmt.Print(1)
		fmt.Print(0)
	}
}

runtime.NumGoroutine()獲取當(dāng)前運行中的goroutine數(shù)量

先介紹一個最簡單的監(jiān)控方式。

通過 runtime.NumGoroutine() 獲取當(dāng)前運行中的 goroutine 數(shù)量,通過它確認是否發(fā)生泄漏。

func main() {
 go test()
 go test()
 go test()
 go test()
 a:=runtime.NumGoroutine()
 fmt.Println(a) // 5
 for  {
 }
}

多任務(wù)資源競爭問題

package main
import (
	"fmt"
	"time"
)
//定義一個打印機,參數(shù)為字符串,按每個字符打印
//打印機屬于公共資源
func Printer(str string) {
	for _, data := range str {
		fmt.Printf("%c", data)
		time.Sleep(time.Second)
	}
	fmt.Printf("\n")
}
func person1() {
	Printer("hello")
}
func person2() {
	Printer("world")
}
func main() {
	//新建2個協(xié)程,代表2個人,2個人同時使用打印機
	go person1()
	go person2()
	//特地不讓主協(xié)程結(jié)束,死循環(huán)
	for {
	}
}

通過channel解決資源競爭問題

package main
import (
	"fmt"
	"time"
)
//全局變量,創(chuàng)建一個channel
var ch = make(chan int)
//定義一個打印機,參數(shù)為字符串,按每個字符打印
//打印機屬于公共資源
func Printer(str string) {
	for _, data := range str {
		fmt.Printf("%c", data)
		time.Sleep(time.Second)
	}
	fmt.Printf("\n")
}
//person1執(zhí)行完后,才能到person2執(zhí)行
func person1() {
	Printer("hello")
	ch <- 666 //給管道寫數(shù)據(jù),發(fā)送
}
func person2() {
	<-ch //從管道取數(shù)據(jù),接收,如果通道沒有數(shù)據(jù)他就會阻塞
	Printer("world")
}
func main() {
	//新建2個協(xié)程,代表2個人,2個人同時使用打印機
	go person1()
	go person2()
	//特地不讓主協(xié)程結(jié)束,死循環(huán)
	for {
	}
}

主協(xié)程如何等其余協(xié)程完再退出

var wg sync.WaitGroup
wg.Add(2)  //任務(wù)數(shù)
wg.Done() //結(jié)束
wg.Wait() //等待
package main
import (
	"fmt"
	"sync"
)
var wg sync.WaitGroup
func main() {
	wg.Add(2)
	out := producer()
	consumer(out)
	defer fmt.Println("主線程結(jié)束")
	wg.Wait()  //等待
}
//此通道只能寫,不能讀
func producer() chan interface{} {
	ch := make(chan interface{})
	go func() {
		for i := 0; i < 5; i++ {
			ch <- fmt.Sprintf("協(xié)程1-%d", i) //寫入字符串
		}
		defer close(ch)
		wg.Done()  //結(jié)束
	}()
	return ch
}
//此channel只能讀,不能寫
func consumer(data chan interface{}) {
	defer fmt.Println("讀取結(jié)束")
	go func() {
		for num := range data {
			fmt.Println(num)
		}
		wg.Done() //結(jié)束
	}()
}

原文鏈接:https://www.cnblogs.com/guyouyin123/p/13985447.html

欄目分類
最近更新