C語言驗(yàn)證大小端的幾個(gè)方法
大小端的問題在很多面試筆試中都會(huì)遇到,最直接的考察是,筆試的時(shí)候,讓你寫一個(gè)代碼,如何確定當(dāng)前系統(tǒng)是大端還是小端的。
什么是大端和小端呢?
大端: 高位字節(jié)排放在內(nèi)存的低地址端,低位字節(jié)排放在內(nèi)存的高地址端。(CPU對操作數(shù)的存放方式是從高字節(jié)到低字節(jié))
小端: 低位字節(jié)排放在內(nèi)存的低地址端,高位字節(jié)排放在內(nèi)存的高地址端。(CPU對操作數(shù)的存放方式是從低字節(jié)到高字節(jié))
假設(shè)我們的內(nèi)存是這樣的
我們要存一個(gè)數(shù)據(jù) 0x44332211到這塊內(nèi)存里面去 如果系統(tǒng)是小端模式的話,存儲(chǔ)方式如下圖
如果系統(tǒng)是大端模式的話,存儲(chǔ)方式如下圖
好了,我們既然知道了大端和小端的存儲(chǔ)方式不同,那就可以寫代碼來判斷當(dāng)前系統(tǒng)的存儲(chǔ)模式了。
方法一
#include <cstdio>
int main()
{
int i = 1;
(*(char *)&i == 1) ? printf("Little-endian\n") : printf("Big-endian\n");
return 0;
}
指針類型轉(zhuǎn)換,最后取 char * 指針的值,也就是判斷 int 低地址的數(shù)據(jù)是否為1。
方法二
#include <cstdio>
union System
{
char a;
int b;
};
int main()
{
union System s;
s.b = 1;
printf("0x%x\n",&s.a);
printf("0x%x\n",&s.b);
if(s.a == 1)
{
printf("Little-endian\n");
}
else
{
printf("Big-endian\n");
}
return 0;
}
共用體的特點(diǎn)是,使用類型最大的那個(gè)類型作為共用體的大小,所以,char a 使用的是 int b的空間大小,判斷 a的值,也就是判斷低地址的數(shù)據(jù)值。
方法三
#include<stdio.h>
static union
{
char a[4];
int ul;
}endian = {{'L', '?', '?', 'B'}};
#define ENDIAN ((char)endian.ul)
int main()
{
printf("%cENDIAN\n", ENDIAN);
return 0;
}
宏和字符數(shù)組
方法四
#include<stdio.h>
int main()
{
int a = 0x44332211;
char *b = (char *)&a;
(*b == 0x11)?printf("Little-endian\n") : printf("Big-endian\n");
return 0;
}
直接指針操作,原理也是取 int 低地址判斷里面的數(shù)據(jù)。
總結(jié)
所有的判斷依據(jù)都是按照上面的理論來驗(yàn)證的,讀者們?nèi)绻麑Υa有什么疑惑或者問題的請給我留言,我也不敢保證自己寫的代碼一定是正確的,幾個(gè)方法也是參考網(wǎng)上的寫法,我最喜歡還是用union來實(shí)現(xiàn)。如果在面試中突然不知道怎么寫,我建議還是畫個(gè)圖出來先理解一下,這樣更有利于寫代碼。
我原來也是上來就想把代碼寫出來,經(jīng)過這些年的錘煉,我明白了,原理遠(yuǎn)比代碼來的重要,我們學(xué)習(xí)也不要局限在某種語言上面,語言是為了實(shí)現(xiàn)某種需求的,C,C++,python,等等,都只是一種語言而已,說白了,就是一把刀,磨得再鋒利的刀,沒有豬肉來砍也只是用來觀賞而已。
本文授權(quán)轉(zhuǎn)載自公眾號(hào)“嵌入式Linux”,作者寫代碼的籃球球癡
免責(zé)聲明:本文內(nèi)容由21ic獲得授權(quán)后發(fā)布,版權(quán)歸原作者所有,本平臺(tái)僅提供信息存儲(chǔ)服務(wù)。文章僅代表作者個(gè)人觀點(diǎn),不代表本平臺(tái)立場,如有問題,請聯(lián)系我們,謝謝!