找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 2212|回復(fù): 0
打印 上一主題 下一主題
收起左側(cè)

關(guān)于C語言算術(shù)表達式中“尋常算術(shù)轉(zhuǎn)換”的問題

[復(fù)制鏈接]
跳轉(zhuǎn)到指定樓層
樓主
ID:75263 發(fā)表于 2015-3-25 00:09 | 只看該作者 回帖獎勵 |倒序瀏覽 |閱讀模式
在c語言算術(shù)表達式的編譯過程中,有一個“尋常算術(shù)轉(zhuǎn)換”的問題,大多數(shù)情況下,教材中并沒有做具體的有意義的說明,在實際應(yīng)用中,如果不注意這個問題,可能會產(chǎn)生嚴(yán)重的后果!一下以實例做一個說明,希望引起大家的注意。
1、C語言算術(shù)表達式“尋常算術(shù)轉(zhuǎn)換”
        由于“歷史”(搞笑,C語言弄出來不過幾十年而已,好意思談歷史?)的原因,C語言算術(shù)表達式在編譯過程中會對參入運算的對象做“尋常算術(shù)轉(zhuǎn)換”,按照ANSIC標(biāo)準(zhǔn)的說法,大致的描述是這樣的:
當(dāng)執(zhí)行算術(shù)運算時,如果操作對象的類型不同,就會發(fā)生數(shù)據(jù)類型轉(zhuǎn)換。對于雙目運算符而言,兩個運算對象的類型都轉(zhuǎn)換為精度高的那個對象的類型。數(shù)據(jù)類型轉(zhuǎn)換的一般規(guī)則是:浮點朝著精度更高、長度更長的方向轉(zhuǎn)換,整型(char,short,int)運算對象如果轉(zhuǎn)換為signed int類型不會丟失信息,就轉(zhuǎn)換為signed int類型,否則轉(zhuǎn)換為 unsigned int類型(詳細(xì)表述請參考ANSIC規(guī)范)。
    嚴(yán)重建議你注意:整型運算對象(char,short,int)的轉(zhuǎn)換規(guī)則!由于在很多系統(tǒng)中,long類型與int類型長度不同,比如在C51中,long是32位的,而int是16位的,就有可能因“想當(dāng)然”而出現(xiàn)錯誤!在C51編譯環(huán)境KEIL里,仿真運行下面的例子,x==?
unsigned long x=0x00123456;
x |= (0xe8<<24);
... ...
想當(dāng)然的結(jié)論:x==0xe8123456
想當(dāng)然的"當(dāng)然":“x |= (0xe8<<24);”等價于“x =x | (0xe8<<24);”,按照“尋常算術(shù)轉(zhuǎn)換”規(guī)則,0xe8被轉(zhuǎn)換為unsigned long類型了,所以x==0xe8123456
但實際上上述結(jié)論是錯誤的。正確的結(jié)論是:x==0x00123456
原因分析:C編譯器首先考慮的是子算術(shù)表達式(子表達式也是表達式)“(0xe8<<24)”!雙目運算符“<<”的兩個運算
對象都是“沒有顯式類型說明的”常數(shù),于是按照“整型(char,short,int)運算對象如果轉(zhuǎn)換為signed int類型不會丟失信息,就轉(zhuǎn)換為signed int類型,否則轉(zhuǎn)換為 unsigned int類型”的規(guī)則,0xe8轉(zhuǎn)換為16位的0000 0000 11101000B,左移24位以后就變成0000 0000 0000 0000B了!
然后這個16位全0的數(shù)與32位的x做 “|”運算, 16位全0轉(zhuǎn)換為24位全0而已!
假定希望把一個字節(jié)常量0xe8“裝配”到32位的x最高8位上,正確的做法應(yīng)該是怎么樣的呢?這樣寫就行了:
unsigned long x=0x00123456;
x |= ((unsigned long)0xe8<<24);

或者,幾乎沒有可移植性的寫法(顯然也就嚴(yán)重的不推薦!。。。
unsigned long x=0x00123456;
x |= (0xppppqqe8<<24);    //0xpppp=0x0001~0xffff, 0xqq=0x00~0xff
說明一下這個嚴(yán)重不推薦的寫法以說明問題的本質(zhì):
0xppppqqe8的形式,最小的數(shù)是0x000100e8,最大的數(shù)是0xffffffe8,不管咋樣,在C51里都必須用32位表示才行,所以<<24不會變成0了!
以下是從前日志里一個實際的例子,用于2M字節(jié)Flash芯片的讀訪問,其中的變量address是unsigned long類型的,它的低端三個字節(jié)用于存放字節(jié)地址,最高端字節(jié)存放“連續(xù)讀”命令0xe8。
void Flash_Read(ulong address, char* buf, uint nbytes)
{
    uchar i;
    AT45161_CLK = 1;
    AT45161_CS(0);
    address |= ((ulong)0xE8<<24);   //注意ulong強制類型轉(zhuǎn)換!
    for(i=0; i<4; i++)                      //4字節(jié)命令/地址
        Flash_Byte_Write((address>>(24-8*i))&0xff);
    for(i=0; i<4; i++) Flash_Byte_Write(0xff); //4字節(jié)填充位
    while(nbytes--) Flash_Byte_Read(buf++);
    AT45161_CS(1);
}
參考文獻:C專家編程(Expert C Programming) 【美】Peter Van Der Linden 人郵出版社

分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏 分享淘帖 頂 踩
回復(fù)

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規(guī)則

手機版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術(shù)交流QQ群281945664

Powered by 單片機教程網(wǎng)

快速回復(fù) 返回頂部 返回列表