1分鐘瞭解C語言正確使用位元組對齊及#pragma pack的方法

来源:https://www.cnblogs.com/Sharemaker/archive/2023/05/05/17375352.html
-Advertisement-
Play Games

​ C/C++編譯器的預設位元組對齊方式為自然對界。即在預設情況下,編譯器為每一個變數或是數據單元按其自然對界條件分配空間。 在結構中,編譯器為結構的每個成員按其自然對界(alignment)條件分配空間。各個成員按照它們被聲明的順序在記憶體中順序存儲(成員之間可能有插入的空位元組),第一個成員的地址和整 ...


​  C/C++編譯器的預設位元組對齊方式為自然對界。即在預設情況下,編譯器為每一個變數或是數據單元按其自然對界條件分配空間。

  在結構中,編譯器為結構的每個成員按其自然對界(alignment)條件分配空間。各個成員按照它們被聲明的順序在記憶體中順序存儲(成員之間可能有插入的空位元組),第一個成員的地址和整個結構的地址相同。

  編譯器預設的結構成員自然對界條件為“N位元組對齊”,N即該成員數據類型的長度。如int型成員的自然對界條件為4位元組對齊,而double類型的結構成員的自然對界條件為8位元組對齊。若該成員的起始偏移不位於該成員的“預設自然對界條件”上,則在前一個節面後面添加適當個數的空位元組。

        編譯器預設的結構整體的自然對界條件為:該結構所有成員中要求的最大自然對界條件。若結構體各成員長度之和不為“結構整體自然對界條件的整數倍”,則在最後一個成員後填充空位元組。

        例子1(分析結構各成員的預設位元組對界條界條件和結構整體的預設位元組對界條件):

1 struct Test
2 { 
3   char x1; // 成員x1為char型(其起始地址必須1位元組對界),其偏移地址為0 
4   char x2; // 成員x2為char型(其起始地址必須1位元組對界,其偏移地址為1 
5   float x3; // 成員x3為float型(其起始地址必須4位元組對界),編譯器在x2和x3之間填充了兩個空位元組,其偏移地址為4 
6   char x4; // 成員x4為char型(其起始地址必須1位元組對界),其偏移地址為8 
7 };

        在Test結構體中,最大的成員為float x3,因此結構體的自然對界條件為4位元組對齊。則結構體長度就為12位元組,記憶體佈局為1100 1111 1000。

 1 #include <stdio.h>
 2 typedef struct
 3 {
 4   int aa1; //4個位元組對齊 1111
 5   char bb1;//1個位元組對齊 1
 6   short cc1;//2個位元組對齊 011
 7   char dd1; //1個位元組對齊 1
 8 } testlength1;
 9 int length1 = sizeof(testlength1); //4個位元組對齊,占用位元組1111 1011 1000,length = 12
10 
11 typedef struct
12 {
13   char bb2;//1個位元組對齊 1
14   int aa2; //4個位元組對齊 01111
15   short cc2;//2個位元組對齊 11
16   char dd2; //1個位元組對齊 1
17 } testlength2;
18 int length2 = sizeof(testlength2); //4個位元組對齊,占用位元組1011  1111 1000,length = 12
19 
20 typedef struct
21 {
22   char bb3; //1個位元組對齊 1
23   char dd3; //1個位元組對齊 1
24   int aa3; //4個位元組對齊 001111
25   short cc23//2個位元組對齊 11
26 
27 } testlength3;
28 int length3 = sizeof(testlength3); //4個位元組對齊,占用位元組1100 1111 1100,length = 12
29 
30 typedef struct
31 {
32   char bb4; //1個位元組對齊 1
33   char dd4; //1個位元組對齊 1
34   short cc4;//2個位元組對齊 11
35   int aa4; //4個位元組對齊 1111
36 } testlength4;
37 int length4 = sizeof(testlength4); //4個位元組對齊,占用位元組1111 1111,length = 8
38 
39 int main(void)
40 {
41   printf("length1 = %d.\n",length1);
42   printf("length2 = %d.\n",length2);
43   printf("length3 = %d.\n",length3);
44   printf("length4 = %d.\n",length4);
45   return 0;
46 }

        改變預設的對界條件(指定對界)
· 使用偽指令#pragma pack (n),編譯器將按照n個位元組對齊。
· 使用偽指令#pragma pack (),取消自定義位元組對齊方式。

        這時,對齊規則為:

        1、數據成員對齊規則:結構(struct)(或聯合(union))的數據成員,第一個數據成員放在offset為0的地方,以後每個數據成員的對齊按照#pragma pack指定的數值和這個數據成員自身長度中,比較小的那個進行。

        2、結構(或聯合)的整體對齊規則:在數據成員完成各自對齊之後,結構(或聯合)本身也要進行對齊,對齊將按照#pragma pack指定的數值和結構(或聯合)最大數據成員長度中,比較小的那個進行。

        結合1、2推斷:當#pragma pack的n值等於或超過所有數據成員長度的時候,這個n值的大小將不產生任何效果。

        因此,當使用偽指令#pragma pack (2)時,Test結構體的大小為8,記憶體佈局為11 11 11 10。

        需要註意一點,當結構體中包含一個子結構體時,子結構中的成員按照#pragma pack指定的數值和子結構最大數據成員長度中,比較小的那個進行進行對齊。例子如下:

 1 #pragma pack(8)
 2 struct s1
 3 {
 4   short a;
 5   long b;
 6 };
 7 
 8 struct s2
 9 {
10   char c;
11   s1 d;
12   long long e;
13 };
14 #pragma pack()

        sizeof(s2)的結果為24。S1的記憶體佈局為1100 1111,S2的記憶體佈局為1000 1100 1111 0000 1111 1111。

        例子2按照2個位元組對齊時:

 1 #include <stdio.h>
 2 #pragma pack(2)
 3 typedef struct
 4 {
 5   int aa1; //2個位元組對齊 1111
 6   char bb1;//1個位元組對齊 1
 7   short cc1;//2個位元組對齊 011
 8   char dd1; //1個位元組對齊 1
 9 } testlength1;
10 int length1 = sizeof(testlength1); //2個位元組對齊,占用位元組11 11 10 11 10,length = 10
11 
12 typedef struct
13 {
14   char bb2;//1個位元組對齊 1
15   int aa2; //2個位元組對齊 01111
16   short cc2;//2個位元組對齊 11
17   char dd2; //1個位元組對齊 1
18 } testlength2;
19 int length2 = sizeof(testlength2); //2個位元組對齊,占用位元組10 11 11 11 10,length = 10
20 
21 typedef struct
22 {
23   char bb3; //1個位元組對齊 1
24   char dd3; //1個位元組對齊 1
25   int aa3; //2個位元組對齊 11 11
26   short cc23//2個位元組對齊 11
27 
28 } testlength3;
29 int length3 = sizeof(testlength3); //2個位元組對齊,占用位元組11 11 11 11,length = 8
30 
31 typedef struct
32 {
33   char bb4; //1個位元組對齊 1
34   char dd4; //1個位元組對齊 1
35   short cc4;//2個位元組對齊 11
36   int aa4; //2個位元組對齊 11 11
37 } testlength4;
38 int length4 = sizeof(testlength4); //2個位元組對齊,占用位元組11 11 11 11,length = 8
39 #pragma pack()
40 int main(void)
41 {
42   printf("length1 = %d.\n",length1);
43   printf("length2 = %d.\n",length2);
44   printf("length3 = %d.\n",length3);
45   printf("length4 = %d.\n",length4);
46   return 0;
47 }

        另外,還有如下的一種方式:

        · __attribute((aligned (n))),讓所作用的結構成員對齊在n位元組自然邊界上。如果結構中有成員的長度大於n,則按照最大成員的長度來對齊。

        · __attribute__ ((packed)),取消結構在編譯過程中的優化對齊,按照實際占用位元組數進行對齊。

       以上的n = 1, 2, 4, 8, 16... 第一種方式較為常見。


↓↓↓更多技術內容和書籍資料獲取,入群技術交流敬請關註“明解嵌入式”↓↓↓ 

 

本文來自博客園,作者:Sharemaker,轉載請註明原文鏈接:https://www.cnblogs.com/Sharemaker/p/17375352.html


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

-Advertisement-
Play Games
更多相關文章
  • 博主給大家推薦一套全部開源的H5電商項目waynboot-mall。由博主在2020年開發至今,已有三年之久。那時候網上很多的H5商城項目都是半開源版本,要麼沒有H5前端代碼,要麼需要加群咨詢,屬實噁心。於是博主決定自己開發一套完整的移動端H5商城,包含一個管理後臺、一個前臺H5商城、一套後端介面。 ...
  • 前言 使用go語言做開發差不多快一年了,主要用來寫後端Web服務,從一開始吐槽他的結構體,比如創建個複雜的JSON格式數據,那是相當的痛苦。還有 err 處理寫的巨麻煩。 當然,go 也有爽的地方,創建個線協程簡直太簡單了。 到後來慢慢接受,覺得效率還行,因為是靜態強類型語言,在修改完項目代碼之後, ...
  • 最近的需求里有這樣一個場景,要校驗一個集合中每個對象的多個Id的有效性。比如一個Customer對象,有3個Id:id1,id2,id3,要把這些Id全部取出來,然後去資料庫里查詢它是否存在。 @Data @AllArgsConstructor public class Customer { pri ...
  • 說明 使用 VLD 記憶體泄漏檢測工具輔助開發時整理的學習筆記。 本篇介紹如何在 VS 高版本中使用 vld2.5.1。同系列文章目錄可見 《記憶體泄漏檢測工具》目錄 1. 使用前的準備 參考本人另一篇博客 安裝 Visual Leak Detector 下載 vld-2.5.1-setup.exe 並 ...
  • 項目背景 隨著互聯網的普及,已經和我們的生活不可分割;寵物漸漸的已經成為了我們的好朋友,寵物醫院管理系統可以幫助用戶合理的管理寵物,呵護寵物的健康,對寵物起到了一個健康監控的作用; 項目介紹 系統總體分為3個角色:分別是系統管理員;醫生和用戶;不能的角色擁有不同的功能許可權,下麵詳細介紹: 系統管理員 ...
  • 作為開發it行業一員,學習借鑒他人項目是很有必要的,所以我們一般都會從github或者 Gitee 上面去參考借鑒他人的項目來學習增加自己的項目經驗 但是github你真的用對了嘛,他的功能其實很強大!!! githu項目搜索 關鍵字搜索 在Github搜索欄中輸入與您感興趣的技術相關的關鍵詞,例如 ...
  • (MyBatis 配置詳解) mybatis-config.xml 核心配置文件 mybatis-config.xml 包含的內容如下 configuration(配置) properties(屬性) settings(設置) typeAliases(類型別名) typeHandlers(類型處理器 ...
  • ##(1) vector:將元素置於一個動態數組中,可以隨機存儲元素(也就是用索引直接存取)。 數組尾部添加或刪除元素非常迅速。但在中部或頭部就比較費時。 *代碼演示:* 取: at在下標越界時會拋出異常,我們能捕獲異常進行處理;而[]下標越界會讓程式直接終止; 構造函數: cbegin, cend ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...