• <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>
            隨筆 - 89  文章 - 118  trackbacks - 0
            <2012年12月>
            2526272829301
            2345678
            9101112131415
            16171819202122
            23242526272829
            303112345

            留言簿(16)

            隨筆分類(56)

            隨筆檔案(89)

            文章分類

            推薦博客

            搜索

            •  

            最新隨筆

            最新評論

            閱讀排行榜

            轉自:http://wenku.baidu.com/view/9e2d2f3e5727a5e9856a6167.html

            大小端問題

             

            By unanao

            <sunjianjiao@gmail.com>

             

            一、什么是大小端問題

            (FromComputer Systems,A Programer's Perspective)在幾乎所有的機器上,多字節(jié)對象被存儲為連續(xù)的字節(jié)序列,對象的地址為所使用字節(jié)序列中最低字節(jié)地址。

            小端:某些機器選擇在存儲器中按照從最低有效字節(jié)到最高有效字節(jié)的順序存儲對象,這種最低有效字節(jié)在最前面的表示方式被稱為小端法(little endian) 。這樣的存儲模式有點兒類似于把數(shù)據(jù)當作字符串順序處理:地址由小向大增加,而數(shù)據(jù)從高位往低位放;

                   大端:某些機器則按照從最高有效字節(jié)到最低有效字節(jié)的順序儲存,這種最高有效字節(jié)在最前面的方式被稱為大端法(big endian) 。這種存儲模式將地址的高低和數(shù)據(jù)位權有效地結合起來,高地址部分權值高,低地址部分權值低,和我們的邏輯方法一致。

             

             舉個例子來說名大小端比如一個int x, 地址為0x100, 它的值為0x1234567. 則它所占據(jù)的0x100, 0x101, 0x102, 0x103地址組織如下圖:




            二、為什么會有大小端模式之分呢?

            這是因為在計算機系統(tǒng)中,我們是以字節(jié)為單位的,每個地址單元都對應著一個字節(jié),一個字節(jié)為 8bit。但是在C語言中除了8bitchar之外,還有16bitshort型,32bitlong型(要看具體的編譯器),另外,對于位數(shù)大于 8位的處理器,例如16位或者32位的處理器,由于寄存器寬度大于一個字節(jié),那么必然存在著一個如果將多個字節(jié)安排的問題。因此就導致了大端存儲模式和小端存儲模式。例如一個16bitshortx,在內存中的地址為0x0010,x的值為0x1122,那么0x11為高字節(jié),0x22為低字節(jié)。對于 大端模式,就將0x11放在低地址中,即0x0010中,0x22放在高地址中,即0x0011中。小端模式,剛好相反。我們常用的X86結構是小端模 式,而KEIL C51則為大端模式。很多的ARM,DSP都為小端模式。有些ARM處理器還可以由硬件來選擇是大端模式還是小端模式。

             

            三、如何區(qū)分大小端問題:

            方法1

            #include <stdio.h>

             

            int main(void)

            {

                   int i = 1;

                   unsigned char *pointer;

             

                   pointer = (unsigned char *)&i;

                   if(*pointer)

                   {

                          printf("litttle_endian");

                   }

                   else

                   {

                          printf("big endian\n");

                   }

             

                   return 0;

            }

                   C中的數(shù)據(jù)類型都是從內存的低地址向高地址擴展,取址運算"&"都是取低地址。小端方式中(i占至少兩個字節(jié)的長度)則i所分配的內存最小地址那個字節(jié)中就存著1,其他字節(jié)是0大端的話則1i的最高地址字節(jié)處存放,char是一個字節(jié),所以強制將char型量p指向ip指向的一定是i的最低地址,那么就可以判斷p中的值是不是1來確定是不是小端。

             

            方法2

            #include <stdio.h>

             

            int main(void)

            {

                   union {

                          short a;

                          char ch;

                   } u;

                   u.a = 1;

             

                   if (u.ch == 1)

                   {

                          printf("Littel endian\n");

                   }

                   else

                   {

                          printf("Big endian\n");

                   }

            }

                   利用聯(lián)合體的特點,數(shù)據(jù)成員共享內存空間,union中元素的起始地址都是相同的——位于聯(lián)合的開始。 char來截取感興趣的字節(jié)。

             

            四、需要考慮大小端(字節(jié)順序)的情況

            1、所寫的程序需要向不同的硬件平臺遷移,說不定哪一個平臺是大端還是小端,為了保證可移植性,一定提前考慮好。

            2. 在不同類型的機器之間通過網(wǎng)絡傳送二進制數(shù)據(jù)時。 一個常見的問題是當小端法機器產(chǎn)生的數(shù)據(jù)被發(fā)送到大端法機器或者反之時,接受程序會發(fā)現(xiàn),字(word)里的字節(jié)(byte)成了反序的。為了避免這類問 題,網(wǎng)絡應用程序的代碼編寫必須遵守已建立的關于字節(jié)順序的規(guī)則,以確保發(fā)送方機器將它的內部表示轉換成網(wǎng)絡標準,而接受方機器則將網(wǎng)絡標準轉換為它的內部標準。

            3. 當閱讀表示整數(shù)的字節(jié)序列時。這通常發(fā)生在檢查機器級程序時,e.g.:反匯編得到的一條指令:
            80483bd: 01 05 64 94 04 08        add %eax, 0x8049464

            3. 當編寫強轉的類型系統(tǒng)的程序時。如寫入的數(shù)據(jù)為u32型,但是讀取的時候卻是char型的。如:0x1234, 大端讀取為12時,小端獨到的是34。

            六、提高程序的可移植性

            使用宏編譯

            #ifdef LITTLE_ENDIAN

            //小端的代碼

            #else

            //大端的代碼

            #endif

             

            七、大、小端之間的轉換

            1、小端轉換為大端

            #include <stdio.h>

             

            void show_byte(char *addr, int len)

            {

                   int i;

             

                   for (i = 0; i < len; i++)

                   {

                          printf("%.2x \t", addr[i]);

                   }

                   printf("\n");

            }

             

            int endian_convert(int t)

            {

                   int result;

                   int i;

             

                   result = 0;

                   for (i = 0; i < sizeof(t); i++)

                   {

                          result <<= 8;

                          result |= (t & 0xFF);

                          t >>= 8;

                   }

             

                   return result;

            }

             

            int main(void)

            {

                   int i;

                   int ret;

             

                   i = 0x1234567;

             

                   show_byte((char *)&i, sizeof(int));

                   ret = endian_convert(i);

                   show_byte((char *)&ret, sizeof(int));

             

                   return 0;

            }

             

            posted on 2012-12-26 16:06 胡滿超 閱讀(909) 評論(0)  編輯 收藏 引用 所屬分類: 算法 、轉載
            老男人久久青草av高清| 精品国产日韩久久亚洲| 色天使久久综合网天天| 热re99久久精品国99热| 亚洲欧美国产精品专区久久 | 久久久久久狠狠丁香| 久久人与动人物a级毛片| 欧美久久久久久精选9999| 国产精品九九九久久九九| 色偷偷久久一区二区三区| A级毛片无码久久精品免费| 三级三级久久三级久久 | 66精品综合久久久久久久| 久久综合给合久久狠狠狠97色| 午夜视频久久久久一区 | 一日本道伊人久久综合影| 久久国产精品波多野结衣AV| 亚洲国产成人久久综合一| 91精品国产91久久| 色综合久久综合网观看| 国产精品免费久久久久久久久| 91精品婷婷国产综合久久 | 久久99精品久久久大学生| 久久久精品国产免大香伊| 亚洲综合精品香蕉久久网| 久久久久久午夜成人影院 | 午夜精品久久久久| 久久精品免费全国观看国产| 久久精品国产亚洲AV香蕉| 久久免费的精品国产V∧| 嫩草影院久久国产精品| 狠狠综合久久综合中文88| 亚洲国产成人久久综合区| 新狼窝色AV性久久久久久| 狠狠色噜噜狠狠狠狠狠色综合久久| 99久久免费国产精精品| 久久精品视频免费| 精品久久亚洲中文无码| 国产精品成人久久久久久久 | 欧美久久一级内射wwwwww.| 日本WV一本一道久久香蕉|