青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

隨筆-145  評論-173  文章-70  trackbacks-0

上面講到了關于pack的內存對齊和計算方法,這里繼續講實現內存對齊的另一種方式:__declspec( align(#) )

__declspec( align(#) )#pragma pack( n )有密切聯系。

當一個變量或結構體同時受兩者影響時,前者的優先級高。

成員的地址決定于前者及后者,其要么是前者的倍數,要么是后者的倍數,要么是成員的大小的倍數,取最小。

結構體最后的大小于前者有關,其要么是前者的倍數,要么是結構體中最大偏移量的倍數,取最大。

要算出最后結果,必須知道兩者的值或缺省值。

 

下面舉一個例子來詳細的分析:

#include <stdio.h>

#include 
"stdafx.h"
#include 
<stdlib.h>
//using namespace std;

#pragma pack( push, 
4 )

__declspec( align(
32) )struct D
{
    
int i1;
    
double d1;
    
int i2;
    
int i3;
};

int main()
{
    cout 
<< "sizeof(int) = "<<sizeof(int<< endl;
    cout 
<< "sizeof(char) = " << sizeof(char<< endl;
    cout 
<< "sizeof(double) = " << sizeof(double<< endl;
    cout 
<< sizeof(D) << endl;
    system(
"PAUSE");
    
return 0;
}

 

這段代碼在VS 2010中的運行結果是,sizeof(D)的大小為32,而在Dev C++C-Free 5.0以及gcc中的結果都似乎20。下面我們來著重講講關于__declspec( align(#) )的用法:

正如前面所說的,當有__declspec( align(#) )pack的時候,__declspec( align(#) )的優先級要高些。所以對于上面這個例子,我們首先來計算出來每一個的大小。

 

1.       成員的地址如何取?

規則:成員的地址要取pack(n)__declspec( align(m) ),以及成員自身大小這三者之間的最小值,也就是,min(n,m,sizeof(成員變量類型)),那么我們可以對每一個結構體的成員都進行分析。

 

第一個為int類型,占據4B,所以地址是[0~3].

第二個為double類型,它的地址要根據min(4,32,sizeof(double))來判斷,所以應該是4的倍數,也就是相鄰著int類型的i1存放。地址是[4~11]

第三個為int類型,占據4B,同樣應該是4的倍數,地址是[12~15].

第四個為int類型,占據4B,地址為[16~19].

 

從而總的地址是從[0~19]連續存放的20個字節,那么是否sizeof(D)的大小就是20呢?

 

經過測試,我們可以看到,在VS 2010中,結果是32why

 

這就要用__declspec( align(#) )來解釋了。也就是下面第二點的內容。

 

2.       結構體最后的大小如何決定?

規則:結構體最后的大小與__declspec( align(m) )有關,其要么是它的倍數,要么是結構體中最大偏移量的倍數,取最大

 

根據這個規則,這里align32,而結構體中最大的是double類型,也就是應該是max(32,8)=32,所以最后結構體的大小應該是32的倍數,而明顯上面我們看到的實際大小是20B,從而需要擴展到32B

 

在這里,就體現了__declspec( align(m) )的強大作用!

 

同樣的,為了體現該語句的作用,我們去掉這個語句,運用我們前面一節內容的知識,來計算并測試sizeof(D),最終不論是在VS 2010還是Dev C++中,運行的結果都是上面我們所預測的20B

 

OK,下面回到最后的疑問,也就是前面我們提出的,為何加入了__declspec( align(m) )語句之后,在DevC++VS 2010的結果不同?

 

實際上,對于這些內存對齊的處理,不同的編譯器可能采取不同的處理,就像前面一節中所說的,我將pack誤用為package,導致根本沒有達到按照我要求的字節對齊的目的,而且編譯器根本不提供任何警告信息。那么,這里合理的解釋是:Dev C++不支持這種用法。

 

通過查閱資料,參照這篇文章【 SSE指令介紹及其CC++應用 】(http://blog.csdn.net/delphihero/archive/2006/09/24/1270069.aspx),我們可以看到作者有這么一段話:

 

接下來我舉一個例子來說明SSE的指令函數是如何使用的,必須要說明的是我以下的代碼都是在VC7.1的平臺上寫的,不保證對其它如Dev-C++Borland C++等開發平臺的完全兼容。

 

這里要注意一下,我使用了__declspec(align(16))做為數組定義的修釋符,這表示該數組是以16字節為邊界對齊的,因為SSE指令只能支持這種格式的內存數據。

  我們在這里看到了SSE算的強大,相信它會成為多媒體程序員手中用來對付無窮盡流媒體數據的一把利劍。我后面還會寫一些關于SSE算法更復雜應用的文章,敬請關注,感謝您抽時間閱讀!

 

從這篇文章我們可以看到,SSE指令集的情況下,在VC 7.1下才支持__declspec(align(16))這種用法,而對于其他平臺不一定有效。而前面我們使用的Dev C++以及C-Free,都是基于g++或者MinGW,不一定會支持這種方式,或者說,不一定按照這種內存對齊的建議來做,也就造成了結果的不同。

 

 

下面我們來繼續探討結構體中有結構體的情況。

 

先看看下面這段代碼:

#include <stdio.h>

#include 
"stdafx.h"
#include 
<stdlib.h>
//using namespace std;

#pragma pack( push, 
4 )

__declspec( align(
32) )struct D
{
    
int i1;
    
double d1;
    
int i2;
    
int i3;
};

__declspec( align(
16) ) struct E
{
     
int i1;
     D m_d;
     
int i2;
};

int main()
{
    cout 
<< "sizeof(int) = "<<sizeof(int<< endl;
    cout 
<< "sizeof(char) = " << sizeof(char<< endl;
    cout 
<< "sizeof(double) = " << sizeof(double<< endl;
    cout 
<< sizeof(D) << endl;
    cout 
<< sizeof(E) << endl;
    system(
"PAUSE");
    
return 0;
}

 

最后運行的結果是sizeof(E)96,為何會是這個結果呢?我們來詳細講解下。

 

對于結構體E,第一個元素為int類型,所以占據[0~3]地址單元。

第二個元素是一個結構體,該結構體由于受上面__declspec( align(32) )的影響,優先級高,所以起始地址是32的倍數,而且大小為32B,從而應該放置在[32~63]單元處。

最后一個是int類型的變量,大小為4,所以應該是4的倍數,地址為[64~67]

 

故結構體E的大小應該是從[0~67],占據68B,而由于前面還有限制__declspec( align(16) ),同時成員變量的最大偏移是sizeof(D)=32,所以我們最后這個結構體的大小應該是他們中最大值的倍數,也就是32的倍數,68向上取32的倍數應該是96.故結果為96.

 

最后仍然是上面平臺的問題,在Dev C++G++下面的結果不同,原因上面解釋了。


MSDN:

The sizeof value for any structure is the offset of the final member, plus that member's size, rounded up to the nearest multiple of the largest member alignment value or the whole structure alignment value, whichever is greater.

中文:

sizeof的結果都是結構體中最后的一個成員變量加上它的大小,再加上一個填充容量(padding),這個填充大小是成員變量最大的一個對齊參數或整個結構體的對齊參數的倍數,取哪個決定于哪個對齊參數較大

 

ms-help://MS.VSCC.v90/MS.MSDNQTR.v90.en/dv_vclang/html/e4209cbb-5437-4b53-b3fe-ac264501d404.htm

ms-help://MS.VSCC.v90/MS.MSDNQTR.v90.en/dv_vclang/html/9cb63f58-658b-4425-ac47-af8eabfc5878.htm




P.S.:上面是關于內存對齊的研究,如有謬誤,歡迎指出!


附參考資料和拓展:

1. #pragma pack :http://blog.sina.com.cn/s/blog_492aa57901008y3h.html
2. #pragma pack( n )和__declspec( align(#) ) 的偏移量計算方法: http://blog.csdn.net/whoismickey/archive/2009/03/28/4032155.aspx
3. #pragma pack(push,1) (pop) :http://blog.csdn.net/jiang1013nan/archive/2009/11/25/4861248.aspx
4. 關于pragma pack的用法(四) C++中的內存對齊問題: http://www.shnenglu.com/xczhang/archive/2007/12/23/39396.html
5. SSE指令介紹及其C、C++應用:http://blog.csdn.net/delphihero/archive/2006/09/24/1270069.aspx
6. c++中__declspec用法總結: http://sealbird.javaeye.com/blog/855096

 


posted on 2011-03-13 22:30 deercoder 閱讀(10041) 評論(5)  編輯 收藏 引用 所屬分類: C++

評論:
# re: 【內存對齊(二)】__declspec( align(#) )的用法和大小計算 2011-03-14 00:21 | fazhang
內存對齊是在筆試題中才開始接觸的。   回復  更多評論
  
# re: 【內存對齊(二)】__declspec( align(#) )的用法和大小計算 2011-03-14 09:39 | 劉暢
@fazhang
是的,面試中也會出現,這個是有感于我在一次面試中的經歷來進行深入研究的。  回復  更多評論
  
# re: 【內存對齊(二)】__declspec( align(#) )的用法和大小計算 2011-03-14 11:42 | Husiwa
VS2008下驗證通過  回復  更多評論
  
# re: 【內存對齊(二)】__declspec( align(#) )的用法和大小計算 2011-03-14 12:19 | 劉暢
@Husiwa
嗯,VS的后續版本估計都會支持。不過VC 6.0還不支持。  回復  更多評論
  
# re: 【內存對齊(二)】__declspec( align(#) )的用法和大小計算 2012-02-23 23:06 | 游客
成員的地址決定于前者及后者,其要么是前者的倍數,要么是后者的倍數,要么是成員的大小的倍數,取最小。

這個不太對。成員地址跟結構提前的__declspec( align(#) )沒關系。

另外,MSDN中說到__declspec(align( # )) can only increase alignment restrictions.所以__declspec(align( # ))是不可能和別的來取最小值,只可能往大的取  回復  更多評論
  
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            亚洲午夜精品17c| 中国成人在线视频| 99精品视频免费| 欧美日韩在线看| 裸体一区二区三区| 亚洲一区二区伦理| 亚洲另类自拍| 免播放器亚洲| 久久免费视频这里只有精品| 欧美三级日韩三级国产三级| 亚洲福利专区| 在线观看日韩欧美| 久久gogo国模啪啪人体图| 午夜精品久久久久| 欧美日韩妖精视频| 亚洲免费观看| 夜夜嗨av一区二区三区四季av | 久久免费视频网站| 久久精品五月| 国产精品一级在线| 久久久久高清| 亚洲欧美综合国产精品一区| 国产精品毛片a∨一区二区三区|国| 亚洲国产欧美一区二区三区同亚洲| 国产又爽又黄的激情精品视频| 亚洲欧美日韩天堂一区二区| 午夜激情久久久| 国产伦精品一区二区三区免费 | 一本色道久久88亚洲综合88| 噜噜噜躁狠狠躁狠狠精品视频| 久久这里只有| 永久久久久久| 久久综合色综合88| 亚洲激情视频网| 一区二区三区精密机械公司| 欧美另类极品videosbest最新版本| 亚洲高清在线| 亚洲一区二区三区高清不卡| 国产精品高清网站| 亚洲午夜精品网| 国产精品日韩在线一区| 久久国产精品黑丝| 欧美精品xxxxbbbb| 亚洲欧洲日产国产综合网| 99视频一区二区| 国产精品毛片高清在线完整版| 亚洲伊人网站| 久久视频精品在线| 亚洲日本va午夜在线影院| 欧美日韩国产欧美日美国产精品| 一区二区三区国产在线| 欧美在线观看视频一区二区| 一区二区三区我不卡| 欧美激情小视频| 亚洲免费一在线| 蜜桃av久久久亚洲精品| 99视频一区| 国产午夜精品理论片a级探花 | 亚洲欧美不卡| 久久国产精品黑丝| 亚洲第一网站免费视频| 亚洲一区二区四区| 伊人成人开心激情综合网| 欧美国产免费| 午夜欧美电影在线观看| 亚洲第一福利视频| 午夜精品在线| 亚洲毛片一区二区| 国产一区二区按摩在线观看| 欧美国产亚洲视频| 欧美一区二区国产| 亚洲精品自在久久| 欧美成人激情在线| 欧美亚洲日本国产| 日韩视频一区二区三区在线播放免费观看| 国产精品午夜视频| 欧美破处大片在线视频| 久久久国产精品一区二区中文| 日韩一级欧洲| 欧美成人一品| 久久精品二区亚洲w码| 日韩一区二区高清| 亚洲国产精品成人一区二区 | 夜夜嗨av色综合久久久综合网| 久久人人爽人人| 亚洲一区欧美一区| 夜夜爽99久久国产综合精品女不卡| 国产主播一区二区| 国产精品美女久久| 欧美日韩一区二区国产| 免费欧美网站| 久久天天躁夜夜躁狠狠躁2022| 亚洲一区二区精品在线观看| 亚洲老板91色精品久久| 欧美激情久久久| 欧美成人69| 久久综合激情| 久久久久女教师免费一区| 欧美一二三区在线观看| 亚洲专区欧美专区| 亚洲视频网站在线观看| 在线亚洲激情| 在线天堂一区av电影| 亚洲最新色图| 一区二区动漫| 亚洲视频第一页| 一区二区三区久久精品| 99一区二区| 中文在线资源观看网站视频免费不卡 | 嫩草国产精品入口| 久久综合狠狠综合久久综合88| 久久er精品视频| 久久久久久久综合日本| 久久免费视频网| 美女精品一区| 欧美精品久久久久a| 欧美精品成人| 欧美性生交xxxxx久久久| 国产精品国产馆在线真实露脸| 欧美性猛交99久久久久99按摩| 国产精品狠色婷| 国产无遮挡一区二区三区毛片日本| 国产亚洲精品aa午夜观看| 狠狠色综合一区二区| 亚洲国产精品一区二区www| 亚洲人在线视频| 亚洲午夜精品久久久久久浪潮 | 亚洲精品影视在线观看| 亚洲激情网站免费观看| 一本久道综合久久精品| 亚洲在线第一页| 欧美一区二区高清| 羞羞色国产精品| 免费日本视频一区| 99热这里只有成人精品国产| 亚洲无线一线二线三线区别av| 久久成人资源| 欧美国产精品人人做人人爱| 欧美肉体xxxx裸体137大胆| 国产欧美短视频| 亚洲国产色一区| 亚洲午夜未删减在线观看| 久久九九久精品国产免费直播| 暖暖成人免费视频| 一本一本久久a久久精品综合妖精 一本一本久久a久久精品综合麻豆 | 91久久夜色精品国产九色| 亚洲视频播放| 免费观看成人| 一区二区三区四区五区在线 | 欧美剧在线观看| 一本色道久久88综合日韩精品 | 日韩亚洲不卡在线| 亚洲日本视频| 国产区二精品视| 麻豆成人av| 欧美国产日本高清在线| 91久久国产自产拍夜夜嗨| 一本色道久久综合亚洲精品高清 | 国产精品久久毛片a| 国产亚洲毛片在线| 在线亚洲欧美视频| 久久婷婷综合激情| 亚洲一卡久久| 欧美精品91| 亚洲电影免费在线观看| 亚洲欧美亚洲| 亚洲乱码国产乱码精品精天堂| 久久精品夜夜夜夜久久| 国产九九精品视频| 亚洲五月婷婷| 亚洲激情av| 可以免费看不卡的av网站| 国产亚洲欧洲一区高清在线观看| 在线视频你懂得一区二区三区| 亚洲第一区在线观看| 久久久久久久999精品视频| 国产精品视频区| 亚洲永久免费| 99国产精品视频免费观看一公开| 欧美成人精品| 亚洲国产一区二区三区a毛片| 久久婷婷国产综合国色天香| 亚洲欧美日韩中文播放| 国产精品视频免费观看www| 亚洲香蕉在线观看| 一本久道久久综合狠狠爱| 欧美精品一区二区在线播放| 亚洲国产精品传媒在线观看| 免费看精品久久片| 久久久久久穴| 在线精品亚洲| 免费亚洲网站| 麻豆精品一区二区综合av| 1024精品一区二区三区| 牛人盗摄一区二区三区视频| 久久视频免费观看| 亚洲黄网站在线观看| 亚洲国产精品va在线看黑人动漫 | 午夜天堂精品久久久久| 国产日韩精品久久|