如何讓你的結構體更高效

来源:https://www.cnblogs.com/lianshuiwuyi/archive/2023/06/06/17460501.html
-Advertisement-
Play Games

> 文中所涉及到的代碼運行結果均是在64位機器上執行得到的. ## 基礎知識回顧 在Go中,我們可以使用`unsafe.Sizeof(x)`來查看變數所占的記憶體大小。以下是Go內置的數據類型占用的記憶體大小: | 類型 | 記憶體大小(位元組數) | | : | : | | bool | 1 | | in ...


文中所涉及到的代碼運行結果均是在64位機器上執行得到的.

基礎知識回顧

在Go中,我們可以使用unsafe.Sizeof(x)來查看變數所占的記憶體大小。以下是Go內置的數據類型占用的記憶體大小:

類型 記憶體大小(位元組數)
bool 1
int8/uint8 1
int/uint 8
int32/uint32 4
int64/uint64 8
float32 4
float64 8
complex64 8
complex128 16
指針類型:*T, map,func,chan 8
string 16
interface 16
[]T 24

func main() {
	fmt.Println(unsafe.Sizeof(true))     // 1
	fmt.Println(unsafe.Sizeof(int8(1)))  // 1
	fmt.Println(unsafe.Sizeof(int(1)))   // 8
	fmt.Println(unsafe.Sizeof(int32(1))) // 4
	fmt.Println(unsafe.Sizeof(int64(1))) // 8

	fmt.Println(unsafe.Sizeof(float32(1.0))) // 4
	fmt.Println(unsafe.Sizeof(float64(1.0))) // 8

	a := int(1)
	fmt.Println(unsafe.Sizeof(&a)) // 8

	s := "1234"
	fmt.Println(unsafe.Sizeof(s)) //16

	var b interface{}
	fmt.Println(unsafe.Sizeof(b)) //16

	fmt.Println(unsafe.Sizeof([]string{})) // 24
	fmt.Println(unsafe.Sizeof([]int{}))    // 24
}

簡單示例

對於一個結構體,其占用的記憶體大小應該是其內部多個基礎類型占用記憶體大小之和。但實際情況並非如此,甚至欄位順序不同,結構體的大小也不同:

type Example1 struct {
	a int32 // 4
	b int32 // 4
	c int64 // 8
}

type Example2 struct {
	a int32 // 4
	c int64 // 8
	b int32 // 4
}

type Example3 struct {
	a bool   // 1
	b int8   // 1
	c string // 16
}

func main() {
	fmt.Println(unsafe.Sizeof(Example1{})) // 16
	fmt.Println(unsafe.Sizeof(Example2{})) // 24
    fmt.Println(unsafe.Sizeof(Example3{})) // 24,並不是1+1+16=18
}

為什麼會出現上面的情況呢?這就引出了本文的重點:記憶體對齊

記憶體對齊

記憶體對齊(Memory Alignment)是指數據在電腦記憶體中存儲時按照特定規則對齊到記憶體地址的過程。記憶體對齊是由電腦硬體和操作系統所決定的,它可以提高記憶體訪問效率和系統性能。

在電腦體繫結構中,記憶體是以位元組(byte)為單位進行訪問的。數據類型在記憶體中占用的位元組數可以是不同的,例如,整數可能占用2位元組、4位元組或8位元組,而字元可能只占用1位元組。

記憶體對齊的規則要求變數的地址必須是其數據類型位元組數的整數倍。例如,如果一個變數的數據類型是4位元組(32位),那麼它的起始地址必須是4的倍數。

記憶體對齊的主要目的是優化電腦的記憶體訪問性能。當數據按照對齊要求存儲在記憶體中時,讀取和寫入操作可以更高效地進行。如果數據沒有按照對齊要求存儲,電腦可能需要進行多次記憶體讀取操作來獲取完整的數據,這會增加訪問延遲和降低系統性能。

在編程中,特別是在使用結構體和類的語言中,記憶體對齊是一個重要的概念。編譯器會根據數據類型的對齊要求自動進行記憶體對齊操作,以確保數據存儲的正確性和性能優化。但在某些情況下,可以通過顯式地設置對齊屬性來控制數據的對齊方式,以滿足特定的需求。

需要註意的是,不同的硬體平臺和操作系統可能具有不同的記憶體對齊規則和要求。因此,在開發跨平臺應用程式時,應當考慮到這些差異並遵循適當的記憶體對齊規則。

為什麼需要對齊記憶體

記憶體對齊是為了提高電腦系統的記憶體訪問效率和性能而存在的。以下是幾個需要記憶體對齊的原因:

  1. 硬體要求:許多電腦硬體和體繫結構對記憶體訪問有特定的對齊要求。如果數據沒有按照硬體要求進行對齊,可能會導致訪問錯誤、異常或性能下降。通過滿足硬體對齊要求,可以確保數據能夠按照有效的方式訪問,提高系統的穩定性和性能。
  2. 記憶體訪問效率:當數據按照對齊要求存儲在記憶體中時,電腦系統可以更高效地訪問這些數據。對齊數據可以減少或避免多次記憶體訪問,提高數據的讀取和寫入速度。這對於大量的數據操作和高性能計算非常重要。
  3. 緩存性能:現代電腦系統中通常有多級緩存,而緩存的訪問是以特定塊的方式進行的。記憶體對齊可以確保數據按照緩存塊的大小對齊,使得數據能夠更好地利用緩存,減少緩存未命中和讀取延遲,提高緩存性能。
  4. 結構體和類的記憶體佈局:結構體和類通常包含多個成員變數,這些變數按照一定的順序存儲在記憶體中。記憶體對齊確保結構體和類的成員變數按照正確的順序和對齊要求進行存儲,避免記憶體空洞和訪問錯誤,保證數據的正確性和一致性。

總之,記憶體對齊是為了滿足硬體要求、提高記憶體訪問效率和性能而引入的機制。通過合理地進行記憶體對齊,可以提高系統的穩定性、性能和響應速度,並避免潛在的記憶體訪問問題。

如何對齊記憶體

Go團隊開發了一款名為fieldalignment的工具可以幫助我們解決記憶體對齊的問題。

使用下麵的命令安裝fieldalignment工具:

$ go install golang.org/x/tools/go/analysis/passes/fieldalignment/cmd/fieldalignment@latest

還是以上面的代碼為例,可以執行下麵的命令:

$ fieldalignment main.go  
main.go:14:15: struct of size 24 could be 16  
main.go:20:15: struct with 16 pointer bytes could be 8

也可以使用--fix參數直接修改代碼:

$ fieldalignment --fix main.go

修改後的內容如下:

type Example1 struct {
	a int32 // 4
	b int32 // 4
	c int64 // 8
}

type Example2 struct {
	c int64
	a int32
	b int32
}

type Example3 struct {
	c string
	a bool
	b int8
}

孟斯特

聲明:本作品採用署名-非商業性使用-相同方式共用 4.0 國際 (CC BY-NC-SA 4.0)進行許可,使用時請註明出處。
Author: mengbin
blog: mengbin
Github: mengbin92
cnblogs: 戀水無意



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

-Advertisement-
Play Games
更多相關文章
  • # 演算法 in Golang:Recursion(遞歸) ## 遞歸演算法 ### 場景:在套娃中找到寶石 ### 可以這樣做 - while 沒找到: - if 當前項 is 寶石: - return 寶石 - else if 當前項 is 套娃: - 打開這個套娃 - if 當前項 is 寶石: ...
  • # JVM運行時數據區 ## 簡介 JVM運行時數據區包括:JVM棧(虛擬機棧),堆,方法區,本地方法棧,PC寄存器。大概的劃分就是棧和堆,以及一些其他的結構。重點在JVM棧,堆,方法區。JVM規範指出:方法區在邏輯上屬於堆,但是實際的具體的JVM中並不屬於堆的一部分。 在JVM棧中會發生GC和Er ...
  • 本文使用代碼片段的形式來解釋在 go 語言開發中經常遇到的小功能點,由於本人主要使用 java 開發,因此會與其作比較,希望對大家有所幫助。 ...
  • 基於java的養老院管理系統設計與實現;可適用於敬老院管理系統,老人管理系統,老人信息管理系統,java養老院信息系統,java敬老院信息管理系統; ...
  • 一、開通OpenAI賬號 1.註冊OpenAI賬號 官網地址:https://openai.com/ 註意:提前準備好國外手機號,沒有的話用簡訊平臺購買手機號接收簡訊 2.購買國外手機號 地址:https://tiger-sms.com/ 我用支付寶充值了30多元(起充要30,加上手續費30多有點坑 ...
  • 當《阿裡巴巴Java開發手冊》發佈後,我也是仔細進行了閱讀,想從中找出一些“標準”,讓自己的代碼質量提高。手冊中對 Object 的 equals 方法的使用進行了強制,而且推薦使用 JDK7 中工具類 Objects 的 equals 方法,至此之後我就很少使用 Object.equals() 方... ...
  • # if語句-語法格式 簡單理解if語句之後,我們的if語句語法格式有多種,選擇使用哪種取決於要測試的條件數 # 1.if結構 最簡單的if語句只有一個條件測試和一個代碼塊 其語法格式: ![image](https://img2023.cnblogs.com/blog/3179433/202306 ...
  • #### 使用thymeleaf做html模板,由xhtmlrenderer/flying-saucer-pdf-openpdf將html轉為PDF > LGPL 和 MPL 許可 #### pom.xml引入依賴 ````java org.springframework.boot spring-b ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...