0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內(nèi)不再提示

C/C++編譯器的缺省字節(jié)對齊方式

CHANBAEK ? 來源:明解嵌入式 ? 作者:Sharemaker001 ? 2023-04-15 11:24 ? 次閱讀

C/C++編譯器的缺省字節(jié)對齊方式為自然對界。 即在缺省情況下,編譯器為每一個變量或是數(shù)據(jù)單元按其自然對界條件分配空間。

在結構中,編譯器為結構的每個成員按其自然對界(alignment)條件分配空間。 各個成員按照它們被聲明的順序在內(nèi)存中順序存儲(成員之間可能有插入的空字節(jié)),第一個成員的地址和整個結構的地址相同。

編譯器缺省的結構成員自然對界條件為“N字節(jié)對齊”,N即該成員數(shù)據(jù)類型的長度。 如int型成員的自然對界條件為4字節(jié)對齊,而double類型的結構成員的自然對界條件為8字節(jié)對齊。 若該成員的起始偏移不位于該成員的“默認自然對界條件”上,則在前一個節(jié)面后面添加適當個數(shù)的空字節(jié)。

編譯器缺省的結構整體的自然對界條件為:該結構所有成員中要求的最大自然對界條件。 若結構體各成員長度之和不為“結構整體自然對界條件的整數(shù)倍”,則在最后一個成員后填充空字節(jié)。

例子1(分析結構各成員的默認字節(jié)對界條界條件和結構整體的默認字節(jié)對界條件):

struct Test
{ 
  char x1; // 成員x1為char型(其起始地址必須1字節(jié)對界),其偏移地址為0 
  char x2; // 成員x2為char型(其起始地址必須1字節(jié)對界,其偏移地址為1 
  float x3; // 成員x3為float型(其起始地址必須4字節(jié)對界),編譯器在x2和x3之間填充了兩個空字節(jié),其偏移地址為4 
  char x4; // 成員x4為char型(其起始地址必須1字節(jié)對界),其偏移地址為8 
};

在Test結構體中,最大的成員為float x3,因此結構體的自然對界條件為4字節(jié)對齊。 則結構體長度就為12字節(jié),內(nèi)存布局為1100 1111 1000。

例子2:

#include 
typedef struct
{
  int aa1; //4個字節(jié)對齊 1111
  char bb1;//1個字節(jié)對齊 1
  short cc1;//2個字節(jié)對齊 011
  char dd1; //1個字節(jié)對齊 1
} testlength1;
int length1 = sizeof(testlength1); //4個字節(jié)對齊,占用字節(jié)1111 1011 1000,length = 12


typedef struct
{
  char bb2;//1個字節(jié)對齊 1
  int aa2; //4個字節(jié)對齊 01111
  short cc2;//2個字節(jié)對齊 11
  char dd2; //1個字節(jié)對齊 1
} testlength2;
int length2 = sizeof(testlength2); //4個字節(jié)對齊,占用字節(jié)1011  1111 1000,length = 12


typedef struct
{
  char bb3; //1個字節(jié)對齊 1
  char dd3; //1個字節(jié)對齊 1
  int aa3; //4個字節(jié)對齊 001111
  short cc23//2個字節(jié)對齊 11


} testlength3;
int length3 = sizeof(testlength3); //4個字節(jié)對齊,占用字節(jié)1100 1111 1100,length = 12


typedef struct
{
  char bb4; //1個字節(jié)對齊 1
  char dd4; //1個字節(jié)對齊 1
  short cc4;//2個字節(jié)對齊 11
  int aa4; //4個字節(jié)對齊 1111
} testlength4;
int length4 = sizeof(testlength4); //4個字節(jié)對齊,占用字節(jié)1111 1111,length = 8


int main(void)
{
  printf("length1 = %d.\\n",length1);
  printf("length2 = %d.\\n",length2);
  printf("length3 = %d.\\n",length3);
  printf("length4 = %d.\\n",length4);
  return 0;
}

改變?nèi)笔〉膶鐥l件(指定對界)

· 使用偽指令#pragma pack (n),編譯器將按照n個字節(jié)對齊。
· 使用偽指令#pragma pack (),取消自定義字節(jié)對齊方式。

這時,對齊規(guī)則為:

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

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

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

因此,當使用偽指令#pragma pack (2)時,Test結構體的大小為8,內(nèi)存布局為11 11 11 10。

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

#pragma pack(8)
struct s1
{
  short a;
  long b;
};


struct s2
{
  char c;
  s1 d;
  long long e;
};
#pragma pack()

sizeof(s2)的結果為24。 S1的內(nèi)存布局為1100 1111,S2的內(nèi)存布局為1000 1100 1111 0000 1111 1111。

例子2按照2個字節(jié)對齊時:

#include 
#pragma pack(2)
typedef struct
{
  int aa1; //2個字節(jié)對齊 1111
  char bb1;//1個字節(jié)對齊 1
  short cc1;//2個字節(jié)對齊 011
  char dd1; //1個字節(jié)對齊 1
} testlength1;
int length1 = sizeof(testlength1); //2個字節(jié)對齊,占用字節(jié)11 11 10 11 10,length = 10


typedef struct
{
  char bb2;//1個字節(jié)對齊 1
  int aa2; //2個字節(jié)對齊 01111
  short cc2;//2個字節(jié)對齊 11
  char dd2; //1個字節(jié)對齊 1
} testlength2;
int length2 = sizeof(testlength2); //2個字節(jié)對齊,占用字節(jié)10 11 11 11 10,length = 10


typedef struct
{
  char bb3; //1個字節(jié)對齊 1
  char dd3; //1個字節(jié)對齊 1
  int aa3; //2個字節(jié)對齊 11 11
  short cc23//2個字節(jié)對齊 11


} testlength3;
int length3 = sizeof(testlength3); //2個字節(jié)對齊,占用字節(jié)11 11 11 11,length = 8


typedef struct
{
  char bb4; //1個字節(jié)對齊 1
  char dd4; //1個字節(jié)對齊 1
  short cc4;//2個字節(jié)對齊 11
  int aa4; //2個字節(jié)對齊 11 11
} testlength4;
int length4 = sizeof(testlength4); //2個字節(jié)對齊,占用字節(jié)11 11 11 11,length = 8
#pragma pack()
int main(void)
{
  printf("length1 = %d.\\n",length1);
  printf("length2 = %d.\\n",length2);
  printf("length3 = %d.\\n",length3);
  printf("length4 = %d.\\n",length4);
  return 0;
}

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

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

· attribute ((packed)),取消結構在編譯過程中的優(yōu)化對齊,按照實際占用字節(jié)數(shù)進行對齊。

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

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學習之用,如有內(nèi)容侵權或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 存儲
    +關注

    關注

    13

    文章

    4123

    瀏覽量

    85273
  • C++
    C++
    +關注

    關注

    21

    文章

    2085

    瀏覽量

    73301
  • 編譯器
    +關注

    關注

    1

    文章

    1602

    瀏覽量

    48895
  • 變量
    +關注

    關注

    0

    文章

    607

    瀏覽量

    28257
  • 字節(jié)對齊

    關注

    0

    文章

    5

    瀏覽量

    1504
收藏 人收藏

    評論

    相關推薦

    c語言中的字節(jié)對齊

    缺省情況下,C編譯器為每一個變量或是數(shù)據(jù)單元按其自然對界條件分配空間。
    的頭像 發(fā)表于 12-30 11:03 ?2549次閱讀
    <b class='flag-5'>c</b>語言中的<b class='flag-5'>字節(jié)</b><b class='flag-5'>對齊</b>

    SEGGER編譯器優(yōu)化和安全技術介紹 支持最新CC++語言

    SEGGER編譯器是專門為ARM和RISC-V微控制設計的優(yōu)化C/C++編譯器。它建立在強大的Clang前端上,支持最新的
    的頭像 發(fā)表于 06-04 15:31 ?1154次閱讀
    SEGGER<b class='flag-5'>編譯器</b>優(yōu)化和安全技術介紹 支持最新<b class='flag-5'>C</b>和<b class='flag-5'>C++</b>語言

    解決單片機開發(fā)字節(jié)對齊問題的方法

    單片機開發(fā)重點-字節(jié)對齊問題在缺省情況下,C編譯器為每一個變量或是數(shù)據(jù)單元按其自然對界條件分配空間。一般地,可以通過下面的方法來改變
    發(fā)表于 11-22 06:06

    C++編譯器數(shù)值性能比較

    現(xiàn)在市面上,主流的C/C++編譯器包括M$的CL、gcc、Intel的icl、PGI的pgcc及Codegear的bcc(原來屬于Borland公司)。Windows上使用最多的自然是cl,而在更廣闊的平臺上,gcc則是
    發(fā)表于 09-10 11:54 ?8次下載

    Linux下C/C++編譯器gcc使用指南

    1.gcc包含的c/c++編譯器 gcc,cc與c++,g++ gcc和cc是一樣的,c++和g++是一樣的。一般
    發(fā)表于 11-02 10:59 ?0次下載

    TMS320C54x匯編語言工具CC++編譯器的功能優(yōu)化詳細概述

    該系列是一套軟件開發(fā)工具的支持,其中包括一個優(yōu)化的CC++編譯器、匯編、鏈接,以及組合工具。本章提供了這些工具的概述,介紹了功能優(yōu)化的
    發(fā)表于 04-27 09:43 ?10次下載
    TMS320<b class='flag-5'>C</b>54x匯編語言工具<b class='flag-5'>C</b>/<b class='flag-5'>C++</b><b class='flag-5'>編譯器</b>的功能優(yōu)化詳細概述

    MSP430優(yōu)化C/C++編譯器V 3.2用戶指南

    MSP430是由一套軟件開發(fā)工具的支持,其中包括一個優(yōu)化的CC++編譯器、匯編、鏈接、及配套的公用設施。
    發(fā)表于 05-07 17:25 ?9次下載
    MSP430優(yōu)化<b class='flag-5'>C</b>/<b class='flag-5'>C++</b><b class='flag-5'>編譯器</b>V 3.2用戶指南

    MATLAB 64位C語言和C++編譯器應用程序免費下載

    本文檔的主要內(nèi)容詳細介紹的是MATLAB 64位C語言和C++編譯器應用程序免費下載。
    發(fā)表于 05-21 08:00 ?4次下載
    MATLAB 64位<b class='flag-5'>C</b>語言和<b class='flag-5'>C++</b><b class='flag-5'>編譯器</b>應用程序免費下載

    阿里云基礎軟件C/C++編譯器的工作現(xiàn)狀及挑戰(zhàn)

    本文主要詳細介紹了阿里云C、C++編譯器(GCC、LLVM)工作現(xiàn)狀、主流編譯器(GCC、LLVM)開源社區(qū)參與現(xiàn)狀以及多架構(RISC-V)對數(shù)據(jù)中心的挑戰(zhàn)。
    的頭像 發(fā)表于 06-23 15:26 ?2821次閱讀
    阿里云基礎軟件<b class='flag-5'>C</b>/<b class='flag-5'>C++</b><b class='flag-5'>編譯器</b>的工作現(xiàn)狀及挑戰(zhàn)

    單片機開發(fā)重點-字節(jié)對齊問題

    單片機開發(fā)重點-字節(jié)對齊問題在缺省情況下,C編譯器為每一個變量或是數(shù)據(jù)單元按其自然對界條件分配空間。一般地,可以通過下面的方法來改變
    發(fā)表于 11-13 13:06 ?6次下載
    單片機開發(fā)重點-<b class='flag-5'>字節(jié)</b><b class='flag-5'>對齊</b>問題

    C編譯器缺省字節(jié)對齊方式(自然對界)

    缺省情況下,C編譯器為每一個變量或是數(shù)據(jù)單元按其自然對界條件分配空間。
    的頭像 發(fā)表于 07-29 09:27 ?2013次閱讀

    SuperH C/C++ 編譯器包 V.9.04 用戶手冊

    SuperH C/C++ 編譯器包 V.9.04 用戶手冊
    發(fā)表于 01-12 18:45 ?1次下載
    SuperH <b class='flag-5'>C</b>/<b class='flag-5'>C++</b> <b class='flag-5'>編譯器</b>包 V.9.04 用戶手冊

    SuperH C/C++ 編譯器包 V.9.01 用戶手冊

    SuperH C/C++ 編譯器包 V.9.01 用戶手冊
    發(fā)表于 04-21 19:55 ?0次下載
    SuperH <b class='flag-5'>C</b>/<b class='flag-5'>C++</b> <b class='flag-5'>編譯器</b>包 V.9.01 用戶手冊

    SuperH C/C++ 編譯器包 V.9.04 用戶手冊

    SuperH C/C++ 編譯器包 V.9.04 用戶手冊
    發(fā)表于 07-03 18:38 ?4次下載
    SuperH <b class='flag-5'>C</b>/<b class='flag-5'>C++</b> <b class='flag-5'>編譯器</b>包 V.9.04 用戶手冊

    RX系列V3.06.00的C/C++編譯器包數(shù)據(jù)手冊

    電子發(fā)燒友網(wǎng)站提供《RX系列V3.06.00的C/C++編譯器包數(shù)據(jù)手冊.pdf》資料免費下載
    發(fā)表于 01-26 15:57 ?1次下載
    RX系列V3.06.00的<b class='flag-5'>C</b>/<b class='flag-5'>C++</b><b class='flag-5'>編譯器</b>包數(shù)據(jù)手冊