• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            Merlin

            Life was like a box of chocolates. You never know what you're gonna get.

               :: 首頁 :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理 ::
              34 隨筆 :: 0 文章 :: 40 評(píng)論 :: 0 Trackbacks

            老師提出一個(gè)很有意思的問題:

            class b
            {
                
            char c1;
                
                
            long l;
                
                
            char c2;
                
            double d;

            }
            ;
            class b
            {
                
            char c1;
                
            char c2;

                    
            long l;
                
                
            double d;
                
            }
            ;
            在存儲(chǔ)的時(shí)候用的空間是不一樣大的,前者是24,后者是16,為什么?
            -------------------------------------------------------------------------------------------------------------------------
            我在自己的機(jī)器上也運(yùn)行了,的確如此,我知道16是怎么來的, char是2,long是4,double是8,2+2+4+8=16,而24我就太明白了,說地址是4字節(jié)的,char是2,要空兩個(gè)字節(jié),那第一個(gè)也應(yīng)該是20阿,怎么會(huì)是24呢?有明白人的給講講!
            posted on 2005-11-11 12:56 Merlin 閱讀(2015) 評(píng)論(15)  編輯 收藏 引用 所屬分類: C++

            評(píng)論

            # re: 一個(gè)有意思的問題 2005-11-11 13:32 笨笨
            你的程序是按照8個(gè)字節(jié)對(duì)齊的
              回復(fù)  更多評(píng)論
              

            # re: 一個(gè)有意思的問題 2005-11-11 14:00 nacci
            建議你為每個(gè)類定義構(gòu)造函數(shù),然后分別生成對(duì)象后,觀察每個(gè)對(duì)象的內(nèi)存布局。在b中,為了和double d對(duì)齊,為c2保留了8 bytes的空間。  回復(fù)  更多評(píng)論
              

            # re: 一個(gè)有意思的問題 2005-11-11 16:37 Merlin
            為什么就char要對(duì)齊阿,而long不用對(duì)齊阿?

            class b
            {
            char c1;

            long l;

            char c2;

            long l1;

            double d;


            };//sizeof(b)=24!
            這里面到底是怎么回事啊?  回復(fù)  更多評(píng)論
              

            # re: 一個(gè)有意思的問題 2005-11-11 16:56 shootingstars
            字節(jié)對(duì)齊是為了提高從內(nèi)存中獲取變量的效率。
            如果數(shù)據(jù)總線的寬度是32位,那么每次從內(nèi)存中取數(shù)據(jù)都是從4的倍數(shù)取的。如果不對(duì)齊的話,有可能取一個(gè)int型數(shù)據(jù)需要兩次操作。

            編譯器一般都可以使用編譯指令來控制是否需要字節(jié)對(duì)齊。  回復(fù)  更多評(píng)論
              

            # re: 一個(gè)有意思的問題 2005-11-11 17:00 笨笨
            如果你是VC的話,使用
            #pragma pack(push,8)
            #pragma pack(1)//使用一個(gè)字節(jié)對(duì)齊
            你在這里寫代碼
            #pragma pack(pop,8)

            這種情況不能亂用,因?yàn)闀?huì)降低效率
            WINDOWS核心編程介紹字節(jié)對(duì)齊的原因  回復(fù)  更多評(píng)論
              

            # re: 一個(gè)有意思的問題 2005-11-11 21:44 Merlin
            謝謝了!  回復(fù)  更多評(píng)論
              

            # re: 一個(gè)有意思的問題 2005-11-12 12:30 Merlin
            當(dāng)CPU的訪問正確對(duì)齊的數(shù)據(jù)時(shí),它的運(yùn)行效率最高,當(dāng)數(shù)據(jù)大小的數(shù)據(jù)模數(shù)的內(nèi)存地址是0時(shí),數(shù)據(jù)是對(duì)齊的。例如:WORD值應(yīng)該是總是從被2除盡的地址開始,而DWORD值應(yīng)該總是從被4除盡的地址開始,數(shù)據(jù)對(duì)齊不是內(nèi)存結(jié)構(gòu)的一部分,而是CPU結(jié)構(gòu)的一部分。當(dāng)CPU試圖讀取的數(shù)據(jù)值沒有正確的對(duì)齊時(shí),CPU可以執(zhí)行兩種操作之一:產(chǎn)生一個(gè)異常條件;執(zhí)行多次對(duì)齊的內(nèi)存訪問,以便讀取完整的未對(duì)齊的數(shù)據(jù),若多次執(zhí)行內(nèi)存訪問,應(yīng)用程序的運(yùn)行速度就會(huì)慢。在最好的情況下,是兩倍的時(shí)間,有時(shí)更長。
            -------------《Windows核心編程》
            不過我還有一點(diǎn)不明白的:
            struct s
            {

            char a;

            long int d;

            double c;


            };//此時(shí)是16
            struct s
            {

            char a;

            long int d;

            double c;

            char a1;


            };//而此時(shí)是24
            也就是說a1用了8個(gè),我覺得就沒有道理啊,不是當(dāng)數(shù)據(jù)大小的數(shù)據(jù)模數(shù)的內(nèi)存地址是0時(shí),數(shù)據(jù)是對(duì)齊的么? char沒有必要用8個(gè)阿!有沒有人明白,幫解釋一下!  回復(fù)  更多評(píng)論
              

            # re: 一個(gè)有意思的問題 2005-11-12 21:21 笨笨
            哎,編譯器默認(rèn)是8個(gè)行不行呀?  回復(fù)  更多評(píng)論
              

            # re: 一個(gè)有意思的問題 2005-11-12 22:49 fireup
            編譯器默認(rèn)的對(duì)齊是8字節(jié)對(duì)齊。

            兩個(gè)結(jié)構(gòu)在內(nèi)存中的分配分別如下圖
            0 4 8
            +-------+-------+
            |char |long | 0-7
            +-------+-------+
            |char | 8-15
            +---------------+
            |double | 16-23
            +---------------+

            0 4 8
            +-------+-------+
            |c|c| |long | 0-7
            +-------+-------+
            |double | 8-15
            +---------------+

              回復(fù)  更多評(píng)論
              

            # re: 一個(gè)有意思的問題 2005-11-13 11:10 味全每日C++
            還是應(yīng)該注意這一點(diǎn) #pragma pack(8)
            默認(rèn)32位是8 ,不排除有的時(shí)候讓你計(jì)算 #pragma pack(4) 的情況..  回復(fù)  更多評(píng)論
              

            # re: 一個(gè)有意思的問題 2005-11-27 02:09
            學(xué)習(xí)  回復(fù)  更多評(píng)論
              

            # re: 一個(gè)有意思的問題 2005-12-30 10:48 發(fā)
            當(dāng)CPU的訪問正確對(duì)齊的數(shù)據(jù)時(shí),它的運(yùn)行效率最高,當(dāng)數(shù)據(jù)大小的數(shù)據(jù)模數(shù)的內(nèi)存地址是0時(shí),數(shù)據(jù)是對(duì)齊的。例如:WORD值應(yīng)該是總是從被2除盡的地址開始,而DWORD值應(yīng)該總是從被4除盡的地址開始,數(shù)據(jù)對(duì)齊不是內(nèi)存結(jié)構(gòu)的一部分,而是CPU結(jié)構(gòu)的一部分。當(dāng)CPU試圖讀取的數(shù)據(jù)值沒有正確的對(duì)齊時(shí),CPU可以執(zhí)行兩種操作之一:產(chǎn)生一個(gè)異常條件;執(zhí)行多次對(duì)齊的內(nèi)存訪問,以便讀取完整的未對(duì)齊的數(shù)據(jù),若多次執(zhí)行內(nèi)存訪問,應(yīng)用程序的運(yùn)行速度就會(huì)慢。在最好的情況下,是兩倍的時(shí)間,有時(shí)更長。
            -------------《Windows核心編程》
            不過我還有一點(diǎn)不明白的:
            struct s
            {

            char a;

            long int d;

            double c;


            };//此時(shí)是16
            struct s
            {

            char a;

            long int d;

            double c;

            char a1;


            };//而此時(shí)是24
            也就是說a1用了8個(gè),我覺得就沒有道理啊,不是當(dāng)數(shù)據(jù)大小的數(shù)據(jù)模數(shù)的內(nèi)存地址是0時(shí),數(shù)據(jù)是對(duì)齊的么? char沒有必要用8個(gè)阿!有沒有人明白,幫解釋一下!
            /////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

            除了結(jié)構(gòu)體內(nèi)的成員變量要求對(duì)齊外,結(jié)構(gòu)自身也是需要對(duì)齊的。結(jié)構(gòu)體成員的對(duì)齊是用成員本身的大小和#pragma pack(push,n)中的n中較小的數(shù)對(duì)齊,例如如果成員大小為2,而你指定的對(duì)齊方式是4,則該成員按2對(duì)齊;結(jié)構(gòu)本身的對(duì)其是用結(jié)構(gòu)中最大成員的大小和#pragma pack(push,n)中的n較小的數(shù)對(duì)齊,例如如果結(jié)構(gòu)中最大成員大小8,而你指定對(duì)齊是16,則結(jié)構(gòu)本身按8對(duì)齊。

            對(duì)于
            struct s
            {

            char a;

            long int d;

            double c;


            };//此時(shí)是16
            編譯器默認(rèn)的一般是8字節(jié)對(duì)齊。
            a大小是1,它就按1字節(jié)對(duì)齊(因?yàn)楸戎付ǖ?小),存儲(chǔ)在0偏移的地方;b大小4,它就按4字節(jié)對(duì)齊(因?yàn)楸戎付ǖ?小),存在偏移4--7的位置,c大小8,它按8字節(jié)對(duì)齊,存在偏移8--15的位置。這樣3個(gè)成員共占用了16字節(jié)。由于該結(jié)構(gòu)最大成員c大小為8,所以結(jié)構(gòu)按8字節(jié)對(duì)齊,16按8園整還是16,因此sizeof s = 16。

            而對(duì)于
            struct s
            {

            char a;

            long int d;

            double c;

            char a1;


            };//而此時(shí)是24
            前3個(gè)成員和上面一樣存儲(chǔ),即a在0位置,d在4--7位置,c在8--16位置,但a1按一字節(jié)對(duì)齊,存在偏移17位置,這樣4個(gè)成員占用了17字節(jié);整個(gè)結(jié)構(gòu)還要按8對(duì)齊(因?yàn)榻Y(jié)構(gòu)最大成員c大小為8),17按8圓整后是24,因此此時(shí)sizeof s = 24。

            如果你指定編譯器按4字節(jié)對(duì)齊,即你在開頭加上#pragma pack(push,4)
            則對(duì)于前者為12,后者為16。
            我用的是vc7.1。





              回復(fù)  更多評(píng)論
              

            # re: 一個(gè)有意思的問題 2005-12-30 11:03 發(fā)
            除了結(jié)構(gòu)體內(nèi)的成員變量要求對(duì)齊外,結(jié)構(gòu)自身也是需要對(duì)齊的。結(jié)構(gòu)體成員的對(duì)齊是用成員本身的大小和#pragma pack(push,n)中的n相比,取較小的數(shù)對(duì)齊,例如如果成員大小為2,而你指定的對(duì)齊方式是4,則該成員按2對(duì)齊;結(jié)構(gòu)本身的對(duì)其是用結(jié)構(gòu)中所有成員對(duì)齊的最大數(shù)和#pragma pack(push,n)中的n相比,取較小的數(shù)對(duì)齊。

            對(duì)于
            struct s
            {

            char a;

            long int d;

            double c;


            };//此時(shí)是16
            編譯器默認(rèn)的一般是8字節(jié)對(duì)齊。
            a大小是1,它就按1字節(jié)對(duì)齊(因?yàn)楸戎付ǖ?小),存儲(chǔ)在0偏移的地方;b大小4,它就按4字節(jié)對(duì)齊(因?yàn)楸戎付ǖ?小),存在偏移4--7的位置,c大小8,它按8字節(jié)對(duì)齊,存在偏移8--15的位置。這樣3個(gè)成員共占用了16字節(jié)。由于該結(jié)構(gòu)成員c對(duì)齊方式為8,是所有成員中最大的,所以結(jié)構(gòu)按8字節(jié)對(duì)齊,16按8園整還是16,因此sizeof s = 16。

            而對(duì)于
            struct s
            {

            char a;

            long int d;

            double c;

            char a1;


            };//而此時(shí)是24
            前3個(gè)成員和上面一樣存儲(chǔ),即a在0位置,d在4--7位置,c在8--16位置,但a1按一字節(jié)對(duì)齊,存在偏移17位置,這樣4個(gè)成員占用了17字節(jié);整個(gè)結(jié)構(gòu)還要按8對(duì)齊為結(jié)17按8圓整后是24,因此此時(shí)sizeof s = 24。

            如果你指定編譯器按4字節(jié)對(duì)齊,即你在開頭加上#pragma pack(push,4)
            則對(duì)于前者為12,后者為16。
            我用的是vc7.1。
              回復(fù)  更多評(píng)論
              

            # re: 一個(gè)有意思的問題 2006-04-06 11:42 clever101
            re發(fā):

            照你的說法:
            #pragma pack(push,4)

            struct s1
            {

            char a;

            long int d;

            double c;


            };

            struct s2
            {

            char a;

            long int d;

            double c;

            char a1;


            };

            s1的大小應(yīng)該是16,s2的大小應(yīng)該是20。事實(shí)上也是這樣的。我在VC上作了測試。我的系統(tǒng)環(huán)境:win2000,vc6.0。  回復(fù)  更多評(píng)論
              

            # re: 一個(gè)有意思的問題 2006-06-30 16:38 cn
            對(duì)階!  回復(fù)  更多評(píng)論
              

            www.久久99| 无码人妻少妇久久中文字幕蜜桃| 丰满少妇人妻久久久久久| 99精品久久精品一区二区| 中文精品久久久久人妻| 久久中文字幕人妻熟av女| 少妇久久久久久久久久| 无码超乳爆乳中文字幕久久| 2021久久国自产拍精品| 国产福利电影一区二区三区,免费久久久久久久精 | 思思久久精品在热线热| 狠狠色丁香久久婷婷综合蜜芽五月| 午夜精品久久影院蜜桃| 亚洲午夜久久久影院| 久久精品a亚洲国产v高清不卡 | 999久久久无码国产精品| 国产精品岛国久久久久| 精品久久人人爽天天玩人人妻| 久久国产免费| AV无码久久久久不卡蜜桃| 99久久99久久精品免费看蜜桃| 久久久久久久尹人综合网亚洲 | 亚洲中文字幕伊人久久无码 | 久久国语露脸国产精品电影| 一本色道久久99一综合| 亚洲国产精品热久久| 一本久久综合亚洲鲁鲁五月天| 少妇久久久久久久久久| 精品久久久无码中文字幕天天| 亚洲精品乱码久久久久久蜜桃图片 | 精品无码久久久久久午夜| 国产无套内射久久久国产| 久久综合偷偷噜噜噜色| 精品亚洲综合久久中文字幕| 91麻豆国产精品91久久久| 久久久久国产精品| 久久99精品国产麻豆宅宅| 99久久国产综合精品网成人影院| 久久久久高潮综合影院| 久久久久婷婷| 99久久精品国产综合一区|