《Go 語言併發之道》讀書筆記(二)

来源:https://www.cnblogs.com/dk168/archive/2022/11/16/16896880.html
-Advertisement-
Play Games

大家好,我是棧長。 今天給大家通報一則框架更新消息,時隔兩個月,Spring Cloud 2021.0.5 最新版發佈了,來看下最新的 Spring Cloud 版本情況: Spring Cloud 無疑是現在 Java 微服務事實上的標準,完全基於 Spring Boot 構建,依賴 Spring ...


今天這篇筆記重點講goroutine

首先怎麼定義goroutine

很簡單,在方法前面加上go就可以了

func main() {
	go sayHello()
}

func sayHello() {
	fmt.Println("hello")
}

也可以直接這樣寫, 基於匿名函數

go func() {
	fmt.Println("hello")
}()

go 語言至少有一個main goroutine, 另外調用的sayhello goroutine和main goroutine併發執行,會在main goroutine退出後退出,所以我們上面的代碼是有問題的,它不會列印出"hello". 因為main goroutine退出了,它沒有機會執行。我們需要用到WaitGroup在main goroutine上面等待它,如下代碼

var wg sync.WaitGroup
wg.Add(1)
go func() {
	defer wg.Done()
	fmt.Println("hello")
}()

wg.Wait()

wg.Add(1), 表示會執行一次, wg.Done就是說執行了一次。 wg.Wait()會等待add的一次done.

main goroutine 可以和 goroutine共用相同的地址空間執行,如下代碼

var wg sync.WaitGroup
salutation := "hello"
wg.Add(1)
go func() {
	defer wg.Done()
	salutation = "welcome"
}()

wg.Wait()
fmt.Println(salutation)

會列印出“welcome”, salutation變數在goroutine中被改變後,也會在main goroutine中生效。

我們來看下麵的代碼會輸出什麼

var wg sync.WaitGroup
for _, salutation := range []string{"hello", "greetings", "good day"} {
	wg.Add(1)
	go func() {
		defer wg.Done()
		mt.Println(salutation)
	}()
}
wg.Wait()

它會列印三個good day, 為什麼呢? 作者的解釋是在goroutine開始之前迴圈有很高的概率會退出,salutation的值不在範圍之內, go 語言運行時會足夠小心的將變數salutation值得引用仍然保留,由記憶體轉移到堆。 我不是特別明白作者得解釋,我自己理解下,感覺是for迴圈是main goroutine, 它執行for很快,調用goroutine得時候它已經迴圈完了,所以就拿到最後得值了。
這個我們想列印三個不同得值,使用下麵得代碼

var wg sync.WaitGroup
for _, salutation := range []string{"hello", "greetings", "good day"} {
	wg.Add(1)
	go func(value string) {
		defer wg.Done()
		fmt.Println(value)
	}(salutation)
}
wg.Wait()

這樣三個goroutine使用的是各自的副本。

goroutine的開銷是什麼樣的呢?

一個goroutine占多少記憶體? 作者書上說大概是2.817KB, 我自己實驗了下我的機器上是9.072KB,而OS線程的一般會是2M, 差距還是有些大,所以我們說啟動百萬的goroutine也是很正常的,而線程一般幾十個就不錯了。 下麵是示例代碼

func main() {
	memConsumed := func() uint64 {
		runtime.GC()
		var s runtime.MemStats
		runtime.ReadMemStats(&s)
		return s.Sys
	}

	var c <-chan interface{}
	var wg sync.WaitGroup
	noop := func() {
		wg.Done()
		<-c

	}

	const numGoroutines = 5e4
	wg.Add(numGoroutines)

	before := memConsumed()

	for i := numGoroutines; i > 0; i-- {
		go noop()
	}

	after := memConsumed()

	fmt.Printf("%.3fkb", float64(after-before)/numGoroutines/1000)
}

numGoroutines是我們想創建的goroutine數量
noop 這個方法一直等channel裡面的value,它不會退出,創建了的goroutine一直在記憶體中。
memConsumed 方法統計當前的記憶體, 我們在開始的時候統計下,在結束的時候統計下,相減後就得到消耗的記憶體。
作者還列舉了goroutine的上下文切換耗時225ns, 而線程切換1467ns, 相差也比較大,

最後

作者得出的結論是使用goroutine非常廉價


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 實現01 1.實現任務階段1 編寫mytomcat,該伺服器能給瀏覽器返回“你好,我是伺服器!”的簡單信息。 根據之前的tomcat框架整體分析,我們將瀏覽器發送請求,tomcat伺服器處理請求,返回資源的整個過程分為三個部分。現在來分析並初步實現第一部分的功能。 1.1基於socket開發服務端流 ...
  • 1.range # 1.range(起始值,結束值,步長) # 範圍 區間 # 步長不寫時預設是1 # 當裡面只有一個數據的時候,預設是0-結束值 # 當裡面有兩個數據的時候,是 起始值-結束值 print('1.range()') print(range(5, 9, 1)) print(type( ...
  • 生成器是一種特殊的迭代器,也可以說其父類型是迭代器 1、迭代器其實也有很多點要清楚,這裡先不做贅述 2、要清楚,迭代器調用iter()返回的是自身 1 def func(s, e): 2 """ 3 生成器函數 4 :param s:起始值 5 :param e: 末值 6 :return: 7 " ...
  • scanf在vs2022中是一個不安全的函數,vs自定義scanf_s是一個安全的函數,但是這在轉給其他編譯器可能不適用 我們可以在開頭加上一個#define _CRT_SECURE_NO_WARNINGS 1可以解決問題 但是自己打比較麻煩,這個時候我們可以打開vs編譯器的文件然後找到VC這個文件 ...
  • 平常是處理文字的時候更多用到的是字元串方面的工具類,作為學習漢字的工具,拼音還是很重要的一個輔助方式,分享一個漢字轉拼音的工具類。可以用於幫助學習漢字或者作為一些漢字的輔助說明都也是可以的。這個還支持將漢字拼音輸出為首字母,例如"愷龍"可以輸出為"kl",在現在大家很多時候都用縮寫的情況下,可以也可 ...
  • 1.HTTP是什麼? http是超文本傳輸協議用來在web瀏覽器和網站伺服器之間傳遞數據信息,http以明文的方式發送內容,不提供任何方式的數據加密,如果攻擊者截獲了Web瀏覽器和網站伺服器之間的傳輸報文,就可以直接讀懂其中的信息,因此,HTTP協議不適合傳輸一些敏感信息,比如:信用卡號、密碼等支付 ...
  • 1、使用註解需要導入的依賴 1、1在application.xml文件中加入該約束 xmlns:context=http://www.springframework.org/schema/context http://www.springframework.org/schema/context ht ...
  • 我國目前並未出台專門針對網路爬蟲技術的法律規範,但在司法實踐中,相關判決已屢見不鮮,K 哥特設了“K哥爬蟲普法”專欄,本欄目通過對真實案例的分析,旨在提高廣大爬蟲工程師的法律意識,知曉如何合法合規利用爬蟲技術,警鐘長鳴,做一個守法、護法、有原則的技術人員。 案情介紹 2018年1月至7月期間,咼某興 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...