• <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>

                  最近調(diào)試網(wǎng)絡(luò)的服務(wù)端程序,自己寫了一個小客戶端程序來測試,發(fā)現(xiàn)服務(wù)程序解包錯誤。經(jīng)調(diào)試發(fā)現(xiàn)客戶端的協(xié)議頭大小和服務(wù)器端的協(xié)議頭大小不一致。原因是服務(wù)器端加了#pragma pack(1),而客戶端沒加。

                之前沒接觸過這個編譯宏,現(xiàn)在來認(rèn)真學(xué)習(xí)之。

                首先google~~

                原來#pragma pack有幾種形式,我所接觸到的是#pragma pack(n),即變量以n字節(jié)對齊。

                變量對齊在每個系統(tǒng)中是不一樣的,默認(rèn)的對齊方式能有效的提高cpu取指取數(shù)的速度,但是可能會浪費一定的空間。在網(wǎng)絡(luò)程序中采用#pragma pack(1),即變量緊縮,不但可以減少網(wǎng)絡(luò)流量,還可以兼容各種系統(tǒng),不會因為系統(tǒng)對齊方式不同而導(dǎo)致解包錯誤。

                了解了概念和優(yōu)點,現(xiàn)在我們就來測試之~

             

                平臺:CPU—Pentium E5700 內(nèi)存—2G

                 1.操作系統(tǒng):ubuntu 11.04 32bit   編譯器:G++ 4.5.2

                 2.操作系統(tǒng):windows xp              編譯器:VS2010

             

                先看第一個測試。

                結(jié)構(gòu)體在正常情況和緊縮情況在以上不同環(huán)境下占用的內(nèi)存大小。

            1 struct pack {
            2   int i;
            3   short s;
            4   double d;
            5   char c;
            6   short f;
            7 }

              
               測試結(jié)果為:

                1
               

             

                2
                
              
                測試結(jié)果分析:

             

             

                可以看出緊縮后結(jié)構(gòu)體的大小為15,是結(jié)構(gòu)體內(nèi)置類型大小的和。但是在默認(rèn)情況下,結(jié)構(gòu)體的大小都是對齊字節(jié)數(shù)的倍數(shù)。ubuntupack只需要20個字節(jié),而windows24個字節(jié)。這是因為ubuntu是以4字節(jié)對齊,而windows則是以最大的內(nèi)置類型的字節(jié)數(shù)對齊,在結(jié)構(gòu)體內(nèi)最大的內(nèi)置類型為double,其大小為8個字節(jié)。他們在內(nèi)存中的對齊方式如下圖:

                1

                

                2

                

               還需注意的是,在對齊類型的內(nèi)部都是以2字節(jié)對齊的。

               結(jié)論:在默認(rèn)情況下,linux操作系統(tǒng)是以4字節(jié)對齊,windows操作系統(tǒng)則是以最大的內(nèi)置類型對齊。

             

               第二個測試

               一個結(jié)構(gòu)體內(nèi)包含另外一個結(jié)構(gòu)體,其大小的情況。

               內(nèi)部的結(jié)構(gòu)體為

            1 struct pack {
            2   short s;
            3   double d;
            4 }

               外部的結(jié)構(gòu)體為

               1 struct complex _pack{

            2   char c;
            3   struct pack s;
            4   double d;
            5 };

                我們有四種情況:

                 1.      pack緊縮,complex _pack緊縮

                 2.      pack緊縮,complex _pack默認(rèn)

                 3.      pack默認(rèn),complex _pack緊縮

                 4.      pack默認(rèn),complex _pack默認(rèn)

                以下的排列均按此順序。

                 測試的結(jié)果

                  1

                  

                 2

                  

                測試結(jié)果分析:

                在兩個操作系統(tǒng)下,除了第一種情況----內(nèi)結(jié)構(gòu)體和外結(jié)構(gòu)體都緊縮----相同之外,其他三種情況都不相同。我們可以根據(jù)偏移畫出結(jié)構(gòu)體在內(nèi)存中的情況。第一種情況省略。

                 1

                 

                 2

                 

                結(jié)論:#pragma pack只影響當(dāng)前結(jié)構(gòu)體的變量的對齊情況,并不會影響結(jié)構(gòu)體內(nèi)部的結(jié)構(gòu)體變量的排列情況。或者說#pragma pack的作用域只是一層。我們由第三種情況,內(nèi)部結(jié)構(gòu)體正常,外部結(jié)構(gòu)體緊縮,可以得出結(jié)構(gòu)體的對齊是按偏移計算的。

                這里還有一個問題沒解決,為什么第二種情況內(nèi)部結(jié)構(gòu)體的偏移都是1?不是4或者8?

             

            posted on 2011-07-15 20:36 Range 閱讀(6228) 評論(6)  編輯 收藏 引用
            評論
            • # re: #pragma pack學(xué)習(xí)
              13174115@qq.com
              Posted @ 2011-07-16 12:50
              樓主你的畫圖工具不錯啊,是什么呢?
                回復(fù)  更多評論   
            • # re: #pragma pack學(xué)習(xí)
              johnnie
              Posted @ 2011-07-20 12:50
              我覺得pack(4)已經(jīng)OK了,沒必要pack(1),畢竟對齊是為了方便自己的CPU。
              要減少網(wǎng)絡(luò)流量的話,可以在發(fā)數(shù)據(jù)前將其壓縮一下,收之前再解壓一次。  回復(fù)  更多評論   
            • # re: #pragma pack學(xué)習(xí)
              Range
              Posted @ 2011-07-21 16:37
              @johnnie

              pack(4)的確是很好的建議,但是具體的情況還需要應(yīng)用。

              IM,bittorrent 這種網(wǎng)絡(luò)程序一般都是IO-bound。CPU的消耗不是很大吧。  回復(fù)  更多評論   
            • # re: #pragma pack學(xué)習(xí)
              Range
              Posted @ 2011-07-21 16:38
              @13174115@qq.com

              visio  回復(fù)  更多評論   
            • # re: #pragma pack學(xué)習(xí)
              劉明杰
              Posted @ 2011-07-21 17:35
              這個15怎么來的,4+2+8+1+2=17,而且我vs2008上跑了也是17。。。。。  回復(fù)  更多評論   
            • # re: #pragma pack學(xué)習(xí)
              Range
              Posted @ 2011-07-22 10:29
              @劉明杰

              謝謝你的提醒,我回頭看了下代碼,pack(1)時候的代碼,沒加short f。

              short f是我后來加上去的,為的是看出在默認(rèn)情況下,內(nèi)部會以2字節(jié)對齊。  回復(fù)  更多評論   

            只有注冊用戶登錄后才能發(fā)表評論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


            統(tǒng)計

            97r久久精品国产99国产精| 精品人妻伦九区久久AAA片69| 久久夜色撩人精品国产| 久久久久亚洲AV成人网| 亚洲乱码中文字幕久久孕妇黑人| 亚洲AV无码成人网站久久精品大| 久久成人国产精品| 欧美国产精品久久高清| 久久狠狠色狠狠色综合| 一本色道久久88综合日韩精品| 日韩精品久久无码人妻中文字幕| 国产精品成人无码久久久久久| 久久热这里只有精品在线观看| 久久综合欧美成人| 一本一道久久综合狠狠老| 久久WWW免费人成—看片| 狠狠色丁香久久综合五月| 久久婷婷色综合一区二区| 精品久久久久久无码中文字幕 | 久久精品国产一区二区电影| 亚洲愉拍99热成人精品热久久| 国内精品久久久久久麻豆| 日本久久久久亚洲中字幕| 久久天天婷婷五月俺也去| 国产成人精品久久亚洲| 久久99国产精品久久99果冻传媒| 精品国产日韩久久亚洲| 日本加勒比久久精品| 久久久精品国产亚洲成人满18免费网站| 久久精品麻豆日日躁夜夜躁| 亚洲欧美伊人久久综合一区二区| 欧美成人免费观看久久| 97香蕉久久夜色精品国产| 久久人人爽人人澡人人高潮AV | 国产精品18久久久久久vr| 无码精品久久久久久人妻中字 | a级毛片无码兔费真人久久| 大伊人青草狠狠久久| 韩国无遮挡三级久久| 情人伊人久久综合亚洲| 国产成人精品久久一区二区三区av|