一区二区三区日韩精品-日韩经典一区二区三区-五月激情综合丁香婷婷-欧美精品中文字幕专区

分享

結(jié)構(gòu)體對齊詳解

 XX_YY_ZZ 2016-04-01
1 -- 結(jié)構(gòu)體數(shù)據(jù)成員對齊的意義
許多實(shí)際的計算機(jī)系統(tǒng)對基本類型數(shù)據(jù)在內(nèi)存中存放的位置有限制,它們會要求這些數(shù)據(jù)的起始地址的值是某個數(shù)k的倍數(shù),這就是所謂的內(nèi)存對齊,而這個k則被稱為該數(shù)據(jù)類型的對齊模數(shù)(alignment modulus)。這種強(qiáng)制的要求一來簡化了處理器與內(nèi)存之間傳輸系統(tǒng)的設(shè)計,二來可以提升讀取數(shù)據(jù)的速度。
比如這么一種處理器,它每次讀寫內(nèi)存的時候都從某個8倍數(shù)的地址開始,一次讀出或?qū)懭?個字節(jié)的數(shù)據(jù),假如軟件能保證double類型的數(shù)據(jù)都從8倍數(shù)地址開始,那么讀或?qū)懸粋€double類型數(shù)據(jù)就只需要一次內(nèi)存操作。否則,我們就可能需要兩次內(nèi)存操作才能完成這個動作,因?yàn)閿?shù)據(jù)或許恰好橫跨在兩個符合對齊要求的8字節(jié)內(nèi)存塊上。
2 -- 結(jié)構(gòu)體對齊包括兩個方面的含義
1)結(jié)構(gòu)體總長度;
2)結(jié)構(gòu)體內(nèi)各數(shù)據(jù)成員的內(nèi)存對齊,即該數(shù)據(jù)成員相對結(jié)構(gòu)體的起始位置;
3 -- 結(jié)構(gòu)體大小的計算方法和步驟
1)將結(jié)構(gòu)體內(nèi)所有數(shù)據(jù)成員的長度值相加,記為sum_a;
2)將各數(shù)據(jù)成員為了內(nèi)存對齊,按各自對齊模數(shù)而填充的字節(jié)數(shù)累加到和sum_a上,記為sum_b。對齊模數(shù)是#pragma pack指定的數(shù)值以及該數(shù)據(jù)成員自身長度中數(shù)值較小者。該數(shù)據(jù)相對起始位置應(yīng)該是對齊模式的整數(shù)倍;
3)將和sum_b向結(jié)構(gòu)體模數(shù)對齊,該模數(shù)是【#pragma pack指定的數(shù)值】、【未指定#pragma pack時,系統(tǒng)默認(rèn)的對齊模數(shù)(32位系統(tǒng)為4字節(jié),64位為8字節(jié))】和【結(jié)構(gòu)體內(nèi)部最大的基本數(shù)據(jù)類型成員】長度中數(shù)值較小者。結(jié)構(gòu)體的長度應(yīng)該是該模數(shù)的整數(shù)倍。
4 -- 結(jié)構(gòu)體大小計算舉例
在計算之前,我們首先需要明確的是各個數(shù)據(jù)成員的對齊模數(shù),對齊模數(shù)和數(shù)據(jù)成員本身的長度以及pragma pack編譯參數(shù)有關(guān),其值是二者中最小數(shù)。如果程序沒有明確指出,就需要知道編譯器默認(rèn)的對齊模數(shù)值。下表是Windows XP/DEV-C++和Linux/GCC中基本數(shù)據(jù)類型的長度和默認(rèn)對齊模數(shù)。
    char short int long float double long long long double
Win-32 長度 1 2 4 4 4 8 8 8
模數(shù) 1 2 4 4 4 8 8 8
Linux-32 長度 1 2 4 4 4 8 8 12
模數(shù) 1 2 4 4 4 4 4 4
Linux-64 長度 1 2 4 8 4 8 8 16
模數(shù) 1 2 4 8 4 8 8 16

例子1:

struct my_struct 
{ 
    char a; 
    long double b; 
};

此例子Windows和Linux計算方法有些許不一致。
在Windows中計算步驟如下:
步驟1:所有數(shù)據(jù)成員自身長度和:1B + 8B = 9B --> sum_a = 9B
步驟2:數(shù)據(jù)成員a放在相對偏移0處,之前不需要填充字節(jié);數(shù)據(jù)成員b為了內(nèi)存對齊,根據(jù)“結(jié)構(gòu)體大小的計算方法和步驟”中第二條原則,其對齊模數(shù)是8,之前需填充7個字節(jié),sum_a + 7 = 16B --> sum_b = 16 B
步驟3:按照定義,結(jié)構(gòu)體對齊模數(shù)是結(jié)構(gòu)體內(nèi)部最大數(shù)據(jù)成員長度和pragma pack中較小者,前者為8后者為4,所以結(jié)構(gòu)體對齊模數(shù)是4。sum_b是4的4倍,不需再次對齊。
綜上3步,可知結(jié)構(gòu)體的長度是16B,各數(shù)據(jù)成員在內(nèi)存中的分布如圖1-1所示。

在Linux中計算步驟如下:
步驟1:所有數(shù)據(jù)成員自身長度和:1B + 12B = 13B --> sum_a = 13B
步驟2:數(shù)據(jù)成員a放在相對偏移0處,之前不需要填充字節(jié);數(shù)據(jù)成員b為了內(nèi)存對齊,根據(jù)“結(jié)構(gòu)體大小的計算方法和步驟”中第二條原則,其對齊模數(shù)是4,之前需填充3個字節(jié),sum_a + 3 = 16B --> sum_b = 16 B
步驟3:按照定義,結(jié)構(gòu)體對齊模數(shù)是結(jié)構(gòu)體內(nèi)部最大數(shù)據(jù)成員長度和pragma pack中較小者,前者為12后者為4,所以結(jié)構(gòu)體對齊模數(shù)是4。sum_b是4的4倍,不需再次對齊。
綜上3步,可知結(jié)構(gòu)體的長度是16B,各數(shù)據(jù)成員在內(nèi)存中的分布如圖1-2所示。

1-1

例子2:

#pragma pack(2) 
struct my_struct 
{ 
    char a; 
    long double b; 
}; 
#pragma pack()

例子1和例子2不同之處在于例子2中使用了#pragma pack(2)編譯參數(shù),它強(qiáng)制指定對齊模數(shù)是2。此例子Windows和Linux計算方法有些許不一致。

在Windows中計算步驟如下:
步驟1:所有數(shù)據(jù)成員自身長度和:1B + 8B = 13B --> sum_a = 9B
步驟2:數(shù)據(jù)成員a放在相對偏移0處,之前不需要填充字節(jié);數(shù)據(jù)成員b為了內(nèi)存對齊,根據(jù)“結(jié)構(gòu)體大小的計算方法和步驟”中第二條原則,其對齊模數(shù)是2,之前需填充1個字節(jié),sum_a + 1 = 10B --> sum_b = 10 B
步驟3:按照定義,結(jié)構(gòu)體對齊模數(shù)是結(jié)構(gòu)體內(nèi)部最大數(shù)據(jù)成員長度和pragma pack中較小者,前者為8后者為2,所以結(jié)構(gòu)體對齊模數(shù)是2。sum_b是2的5倍,不需再次對齊。
綜上3步,可知結(jié)構(gòu)體的長度是10B,各數(shù)據(jù)成員在內(nèi)存中的分布如圖2-1所示。

在Linux中計算步驟如下:
步驟1:所有數(shù)據(jù)成員自身長度和:1B + 12B = 13B --> sum_a = 13B
步驟2:數(shù)據(jù)成員a放在相對偏移0處,之前不需要填充字節(jié);數(shù)據(jù)成員b為了內(nèi)存對齊,根據(jù)“結(jié)構(gòu)體大小的計算方法和步驟”中第二條原則,其對齊模數(shù)是2,之前需填充1個字節(jié),sum_a + 1 = 14B --> sum_b = 14 B
步驟3:按照定義,結(jié)構(gòu)體對齊模數(shù)是結(jié)構(gòu)體內(nèi)部最大數(shù)據(jù)成員長度和pragma pack中較小者,前者為8后者為2,所以結(jié)構(gòu)體對齊模數(shù)是2。sum_b是2的7倍,不需再次對齊。
綜上3步,可知結(jié)構(gòu)體的長度是14B,各數(shù)據(jù)成員在內(nèi)存中的分布如圖2-2所示。

2

例子3:

struct my_struct 
{ 
    char a; 
    double b; 
    char c; 
}; 

前兩例中,數(shù)據(jù)成員在Linux和Windows下都相同,例3中double的對齊模數(shù)在Linux中是4,在Windows下是8,針對這種模數(shù)不相同的情況加以分析。
在Windows中計算步驟如下:
步驟1:所有數(shù)據(jù)成員自身長度和:1B + 8B + 1B = 10B --> sum_a = 10B
步驟2:數(shù)據(jù)成員a放在相對偏移0處,之前不需要填充字節(jié);數(shù)據(jù)成員b為了內(nèi)存對齊,根據(jù)“結(jié)構(gòu)體大小的計算方法和步驟”中第二條原則,其對齊模數(shù)是8,之前需填充7個字節(jié),sum_a + 7 = 17B --> sum_b = 17B
步驟3:按照定義,結(jié)構(gòu)體對齊模數(shù)是結(jié)構(gòu)體內(nèi)部最大數(shù)據(jù)成員長度和pragma pack中較小者,前者為8后者為8,所以結(jié)構(gòu)體對齊模數(shù)是8。sum_b應(yīng)該是8的整數(shù)倍,所以要在結(jié)構(gòu)體后填充8*3 - 17 = 7個字節(jié)。
綜上3步,可知結(jié)構(gòu)體的長度是24B,各數(shù)據(jù)成員在內(nèi)存中的分布如圖3-1所示。

在Linux中計算步驟如下:
步驟1:所有數(shù)據(jù)成員自身長度和:1B + 8B + 1B = 10B,sum_a = 10B
步驟2:數(shù)據(jù)成員a放在相對偏移0處,之前不需要填充字節(jié);數(shù)據(jù)成員b為了內(nèi)存對齊,根據(jù)“結(jié)構(gòu)體大小的計算方法和步驟”中第二條原則,其對齊模數(shù)是4,之前需填充3個字節(jié),sum_b = sum_a + 3 = 13B
步驟3:按照定義,結(jié)構(gòu)體對齊模數(shù)是結(jié)構(gòu)體內(nèi)部最大數(shù)據(jù)成員長度和pragma
pack中較小者,前者為8后者為4,所以結(jié)構(gòu)體對齊模數(shù)是4。sum_b應(yīng)該是4的整數(shù)倍,所以要在結(jié)構(gòu)體后填充4*4 - 13 = 3個字節(jié)。
綜上3步,可知結(jié)構(gòu)體的長度是16B,各數(shù)據(jù)成員在內(nèi)存中的分布如圖3-2所示。

3

例子4:

struct my_struct 
{ 
    char a[11]; 
    int b; 
    char c; 
}; 

此例子Windows和Linux計算方法一樣,如下:
步驟1:所有數(shù)據(jù)成員自身長度和:11B + 4B + 1B = 16B --> sum_a = 16B
步驟2:數(shù)據(jù)成員a放在相對偏移0處,之前不需要填充字節(jié);數(shù)據(jù)成員b為了內(nèi)存對齊,根據(jù)“結(jié)構(gòu)體大小的計算方法和步驟”中第二條原則,其對齊模數(shù)是4,之前需填充3個字節(jié),sum_a + 1 = 17B --> sum_b = 17B
步驟3:按照定義,結(jié)構(gòu)體對齊模數(shù)是結(jié)構(gòu)體內(nèi)部最大數(shù)據(jù)成員長度和pragma pack中較小者,前者為4后者為4,所以結(jié)構(gòu)體對齊模數(shù)是4。sum_b是4的整數(shù)倍,需在結(jié)構(gòu)體后填充4*5 - 17 = 1個字節(jié)。
綜上3步,可知結(jié)構(gòu)體的長度是20B,各數(shù)據(jù)成員在內(nèi)存中的分布如圖4所示。

4

例子5:

struct my_test 
{ 
    int my_test_a; 
    char my_test_b; 
}; 
struct my_struct 
{ 
    struct my_test a; 
    double my_struct_a; 
    int my_struct_b; 
    char my_struct_c; 
}; 

例子5和前幾個例子均不同,在此例子中我們要計算struct my_struct的大小,而my_struct中嵌套了一個my_test結(jié)構(gòu)體。這種結(jié)構(gòu)體應(yīng)該如何計算呢?原則是將my_test在my_struct中先展開,然后再計算,即是展開成如下結(jié)構(gòu)體:

struct my_struct
{
    int my_test_a;
    char my_test_b;
    double my_struct_a;
    int my_struct_b;
    char my_struct_c;
}; 

此例子Windows中的計算方法如下:
步驟1:所有數(shù)據(jù)成員自身長度和:4B + 1B + 8B + 4B + 1B= 18B --> sum_a = 18B
步驟2:數(shù)據(jù)成員my_struct_a為了內(nèi)存對齊,根據(jù)“結(jié)構(gòu)體大小的計算方法和步驟”中第二條原則,其對齊模數(shù)是8,之前需填充3個字節(jié):sum_a + 3 = 21B --> sum_b = 21B
步驟3:按照定義,結(jié)構(gòu)體對齊模數(shù)是結(jié)構(gòu)體內(nèi)部最大數(shù)據(jù)成員長度和pragma pack中較小者,前者為8后者為8,所以結(jié)構(gòu)體對齊模數(shù)是8。sum_b是8的整數(shù)倍,需在結(jié)構(gòu)體后填充3*8 - 21 = 3個字節(jié)。
綜上3步,可知結(jié)構(gòu)體的長度是24B,各數(shù)據(jù)成員在內(nèi)存中的分布如圖5所示。

此例子Linux中的計算方法如下:
步驟1:所有數(shù)據(jù)成員自身長度和:4B + 1B + 8B + 4B + 1B= 18B,sum_a = 18B
步驟2:數(shù)據(jù)成員my_struct_a為了內(nèi)存對齊,根據(jù)“結(jié)構(gòu)體大小的計算方法和步驟”中第二條原則,其對齊模數(shù)是4,之前需填充3個字節(jié),sum_b = sum_a + 3 = 21B
步驟3:按照定義,結(jié)構(gòu)體對齊模數(shù)是結(jié)構(gòu)體內(nèi)部最大數(shù)據(jù)成員長度和pragma
pack中較小者,前者為4后者為4,所以結(jié)構(gòu)體對齊模數(shù)是4。sum_b是4的整數(shù)倍,需在結(jié)構(gòu)體后填充6*4 - 21 = 3個字節(jié)。
綜上3步,可知結(jié)構(gòu)體的長度是24B,各數(shù)據(jù)成員在內(nèi)存中的分布如圖5所示。

5

5 -- 源代碼附錄

上面的例子均在Windows(VC++6.0)和Linux(GCC4.1.0)上測試驗(yàn)證。下面是測試程序。

#include <iostream>

#include <stdio.h>

using namespace std;

int main()
{
    cout << "sizeof(char)        = " << sizeof(char) << endl;
    cout << "sizeof(short)       = " << sizeof(short) << endl;
    cout << "sizeof(int)         = " << sizeof(int) << endl;
    cout << "sizeof(long)        = " << sizeof(long) << endl;
    cout << "sizeof(float)       = " << sizeof(float) << endl;
    cout << "sizeof(double)      = " << sizeof(double) << endl;
    cout << "sizeof(long long)   = " << sizeof(long long) << endl;
    cout << "sizeof(long double) = " << sizeof(long double) << endl << endl;    

    // 例子1
    {
        struct my_struct 
        { 
            char a; 
            long double b; 
        };
        cout << "exapmle-1: sizeof(my_struct) = " << sizeof(my_struct) << endl;
  
        struct my_struct data;
 
        printf("my_struct->a: %u\nmy_struct->b: %u\n\n", &data.a, &data.b);
    }

    // 例子2
    {
        #pragma pack(2) 
        struct my_struct 
        { 
            char a; 
            long double b; 
        }; 
        
        #pragma pack()
        struct my_struct data;
 
        cout << "exapmle-2: sizeof(my_struct) = " << sizeof(my_struct) << endl;
  
        printf("my_struct->a: %u\nmy_struct->b: %u\n\n", &data.a, &data.b);
    }
    
    // 例子3
    {
        struct my_struct 
        { 
            char a; 
            double b; 
            char c; 
        }; 
 
        struct my_struct data;
 
        cout << "exapmle-3: sizeof(my_struct) = " << sizeof(my_struct) << endl;
  
        printf("my_struct->a: %u\nmy_struct->b: %u\nmy_struct->c: %u\n\n", &data.a, &data.b, &data.c);
    }

    // 例子4
    {
        struct my_struct 
        {  
            char a[11];  
            int b;  
            char c;  
        };
  
        cout << "example-4: sizeof(my_struct) = " << sizeof(struct my_struct) << endl;
  
        struct my_struct data;
        printf("my_struct->a: %u\nmy_struct->b: %u\nmy_struct->c: %u\n\n", &data, &data.b, &data.c);
    }

    // 例子5 
    {
        struct my_test 
        { 
            int my_test_a; 
            char my_test_b; 
        }; 
        
        struct my_struct 
        { 
            struct my_test a; 
            double my_struct_a; 
            int my_struct_b; 
            char my_struct_c; 
        }; 
        cout << "example-5: sizeof(my_struct) = " << sizeof(struct my_struct) << endl;
  
        struct my_struct data;
        printf("my_struct->my_test_a  : %u\n"
            "my_struct->my_test_b  : %u\n"
            "my_struct->my_struct_a: %u\n"
            "my_struct->my_struct_b: %u\n"
            "my_struct->my_struct_c: %u\n", &data.a.my_test_a, &data.a.my_test_b, 
            &data.my_struct_a, &data.my_struct_b, &data.my_struct_c);
    }

    return 0;
}
執(zhí)行結(jié)果:
//Linux localhost 3.4.6-2.10-desktop #1 SMP PREEMPT Thu Jul 28 19:20:26 UTC 2012 (641c197) x86_64 x86_64 x86_64 GNU/Linux
sizeof(char)        = 1
sizeof(short)       = 2
sizeof(int)         = 4
sizeof(long)        = 8
sizeof(float)       = 4
sizeof(double)      = 8
sizeof(long long)   = 8
sizeof(long double) = 16

exapmle-1: sizeof(my_struct) = 32
my_struct->a: 2163695552
my_struct->b: 2163695568

exapmle-2: sizeof(my_struct) = 18
my_struct->a: 2163695680
my_struct->b: 2163695682

exapmle-3: sizeof(my_struct) = 24
my_struct->a: 2163695648
my_struct->b: 2163695656
my_struct->c: 2163695664

example-4: sizeof(my_struct) = 20
my_struct->a: 2163695616
my_struct->b: 2163695628
my_struct->c: 2163695632

example-5: sizeof(my_struct) = 24
my_struct->my_test_a  : 2163695584
my_struct->my_test_b  : 2163695588
my_struct->my_struct_a: 2163695592
my_struct->my_struct_b: 2163695600
my_struct->my_struct_c: 2163695604

//Linux localhost 3.4.6-2.10-desktop #1 SMP PREEMPT Thu Jul 26 09:36:26 UTC 2012 (641c197) i686 i686 i386 GNU/Linux
sizeof(char)        = 1
sizeof(short)       = 2
sizeof(int)         = 4
sizeof(long)        = 4
sizeof(float)       = 4
sizeof(double)      = 8
sizeof(long long)   = 8
sizeof(long double) = 12

exapmle-1: sizeof(my_struct) = 16
my_struct->a: 3213889904
my_struct->b: 3213889908

exapmle-2: sizeof(my_struct) = 14
my_struct->a: 3213889890
my_struct->b: 3213889892

exapmle-3: sizeof(my_struct) = 16
my_struct->a: 3213889872
my_struct->b: 3213889876
my_struct->c: 3213889884

example-4: sizeof(my_struct) = 20
my_struct->a: 3213889852
my_struct->b: 3213889864
my_struct->c: 3213889868

example-5: sizeof(my_struct) = 24
my_struct->my_test_a  : 3213889828
my_struct->my_test_b  : 3213889832
my_struct->my_struct_a: 3213889836
my_struct->my_struct_b: 3213889844
my_struct->my_struct_c: 3213889848

//CYGWIN_NT-6.1 motadou-PC 1.7.20(0.266/5/3) 2013-06-07 11:11 i686 Cygwin
sizeof(char)        = 1
sizeof(short)       = 2
sizeof(int)         = 4
sizeof(long)        = 4
sizeof(float)       = 4
sizeof(double)      = 8
sizeof(long long)   = 8
sizeof(long double) = 12

exapmle-1: sizeof(my_struct) = 16
my_struct->a: 2272336
my_struct->b: 2272340

exapmle-2: sizeof(my_struct) = 14
my_struct->a: 2272322
my_struct->b: 2272324

exapmle-3: sizeof(my_struct) = 24
my_struct->a: 2272296
my_struct->b: 2272304
my_struct->c: 2272312

example-4: sizeof(my_struct) = 20
my_struct->a: 2272276
my_struct->b: 2272288
my_struct->c: 2272292

example-5: sizeof(my_struct) = 24
my_struct->my_test_a  : 2272248
my_struct->my_test_b  : 2272252
my_struct->my_struct_a: 2272256
my_struct->my_struct_b: 2272264
my_struct->my_struct_c: 2272268

    本站是提供個人知識管理的網(wǎng)絡(luò)存儲空間,所有內(nèi)容均由用戶發(fā)布,不代表本站觀點(diǎn)。請注意甄別內(nèi)容中的聯(lián)系方式、誘導(dǎo)購買等信息,謹(jǐn)防詐騙。如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,請點(diǎn)擊一鍵舉報。
    轉(zhuǎn)藏 分享 獻(xiàn)花(0

    0條評論

    發(fā)表

    請遵守用戶 評論公約

    類似文章 更多

    国产a天堂一区二区专区| 国产成人免费高潮激情电| 嫩呦国产一区二区三区av| 欧美不雅视频午夜福利| 国产精品偷拍一区二区| 熟女免费视频一区二区| 国产三级不卡在线观看视频| 日韩在线中文字幕不卡| 五月天综合网五月天综合网| 人妻少妇av中文字幕乱码高清| 亚洲av日韩av高潮无打码| 欧美不卡高清一区二区三区| 亚洲国产精品久久网午夜| 色丁香一区二区黑人巨大| 视频一区二区三区自拍偷| 国产男女激情在线视频| 欧美大胆女人的大胆人体| 欧美人禽色视频免费看| 欧美日韩校园春色激情偷拍| 麻豆国产精品一区二区三区| 国产精品视频一区麻豆专区| 国产精品免费不卡视频| 中文字幕欧美精品人妻一区 | 国产三级黄片在线免费看 | 青草草在线视频免费视频 | 国产在线观看不卡一区二区| 欧美成人高清在线播放| 久久大香蕉精品在线观看| 亚洲一区二区福利在线| 国内胖女人做爰视频有没有| 久久碰国产一区二区三区| 久久精品国产在热久久| 激情内射日本一区二区三区| 中文字幕一二区在线观看| 国产又猛又大又长又粗| 欧洲精品一区二区三区四区| 日韩中文字幕在线不卡一区| 国产福利在线播放麻豆| 一区二区三区日韩中文| 熟女中文字幕一区二区三区| 国产传媒高清视频在线|