? 確定一個(gè)變量是有符號(hào)數(shù)還是無符號(hào)數(shù) 收藏
讀《C專家編程》,其中一段講面試,說是微軟曾經(jīng)有一道面試題:
寫一段代碼,確定一個(gè)變量是有符號(hào)數(shù)還是無符號(hào)數(shù)?
書上給出了兩個(gè)宏:
#define ISUNSIGNED(a) (a>=0 && ~a>=0)
#define ISUNSIGNED(type) ((type)0-1 > 0)
第二個(gè)從類型來判斷,沒有問題。
而第一個(gè)只能用在K&R C里,在ANSI C里就不行了。
當(dāng)這個(gè)宏被用在int/unsigned int時(shí),沒有任何問題。
但是當(dāng)使用在char和short上就會(huì)出錯(cuò)。
ANSI C中的整型升級(jí):
char,short int或者int型位段(bit-field),包括它們的有符號(hào)或無符號(hào)變型,
以及枚舉類型,可以使用在需要int或unsigned int的表達(dá)式中,
如果int可以完整地表示源類型的所有值,那么該類型的值就轉(zhuǎn)換為int,否則轉(zhuǎn)換為unsigned int。
ANSI C中的尋常算術(shù)轉(zhuǎn)換:
當(dāng)執(zhí)行算術(shù)運(yùn)算時(shí),操作數(shù)的類型如果不同,就會(huì)發(fā)生轉(zhuǎn)換。
數(shù)據(jù)類型一般朝著浮點(diǎn)精度更高、長度更長的方向轉(zhuǎn)換,
整型數(shù)如果轉(zhuǎn)換為signed不會(huì)丟失信息,就轉(zhuǎn)換為signed,否則就轉(zhuǎn)換為unsigned。
這個(gè)稱為值保留(value preserving)原則。
所以,無論原先是否有符號(hào),char和short都被轉(zhuǎn)換成了signed int(整型升級(jí))。
原先unsigned的東西變成了signed,然后再進(jìn)行取反。
同時(shí),常數(shù)0被認(rèn)為是signed int類型,所以一律被判為有符號(hào)數(shù)了。
問題是一旦char或者short參與了運(yùn)算,它們將被首先轉(zhuǎn)換成int,
在這以后,任何操作都變成徒勞的了,int永遠(yuǎn)都是signed。
那么能否在整型升級(jí)之前讓signed char/short變成負(fù)數(shù)呢?至少我現(xiàn)在還沒想到辦法。
偶使用了賴皮方法,無恥地定義了全局變量,還用了變態(tài)的逗號(hào)表達(dá)式……
于是第一個(gè)宏就變成下面這個(gè)樣子了:
int r, t;
#define ISUNSIGNED(a) (t = a, r = (a>=0 && (a=~a)>=0), a = t, r)
再一想,既然用了全局變量保存a的值,還討論干啥?于是……
int r, t;
#define ISUNSIGNED(a) (t = a, r = ((a=-1) >= 0), a = t, r)
而且這樣做的前提是,假設(shè)int是最長的整型類型,并且是有符號(hào)的。
本文來自CSDN博客,轉(zhuǎn)載請(qǐng)標(biāo)明出處:http://blog.csdn.net/yysdsyl/archive/2007/11/14/1885829.aspx
本文來自CSDN博客,轉(zhuǎn)載請(qǐng)標(biāo)明出處:http://blog.csdn.net/rainbow_free/archive/2009/06/17/4275697.aspx