|
C++位運(yùn)算,看高手都是運(yùn)用的靈活自如,打算從今天開始學(xué)習(xí)他!收藏 每次看到位運(yùn)算的地方,都比較迷糊.以前學(xué)習(xí)C的時(shí)候也不求甚解,到現(xiàn)在看來,覺得位運(yùn)算和指針在C++基本知識(shí)里是最難理解,最難融會(huì)貫通的東西.尤其是位運(yùn)算,用好了可以"出神入化"了^_^. 如果當(dāng)年好好學(xué)習(xí)C語言,也不至于今天這么費(fèi)勁! 位運(yùn)算 位運(yùn)算的運(yùn)算分量只能是整型或字符型數(shù)據(jù),位運(yùn)算把運(yùn)算對(duì)象看作是由二進(jìn)位組成的位串信息,按位完成指定的運(yùn)算,得到位串信息的結(jié)果。 位運(yùn)算符有: &(按位與)、|(按位或)、^(按位異或)、~ (按位取反)。 其中,按位取反運(yùn)算符是單目運(yùn)算符,其余均為雙目運(yùn)算符。 位運(yùn)算符的優(yōu)先級(jí)從高到低,依次為~、&、^、|, 其中~的結(jié)合方向自右至左,且優(yōu)先級(jí)高于算術(shù)運(yùn)算符,其余運(yùn)算符的結(jié)合方向都是自左至右,且優(yōu)先級(jí)低于關(guān)系運(yùn)算符。 (1)按位與運(yùn)算符(&) 按位與運(yùn)算將兩個(gè)運(yùn)算分量的對(duì)應(yīng)位按位遵照以下規(guī)則進(jìn)行計(jì)算: 0 & 0 = 0, 0 & 1 = 0, 1 & 0 = 0, 1 & 1 = 1。 即同為 1 的位,結(jié)果為 1,否則結(jié)果為 0。 例如,設(shè)3的內(nèi)部表示為 00000011 5的內(nèi)部表示為 00000101 則3&5的結(jié)果為 00000001 按位與運(yùn)算有兩種典型用法,一是取一個(gè)位串信息的某幾位,如以下代碼截取x的最低7位:x & 0177。二是讓某變量保留某幾位,其余位置0,如以下代碼讓x只保留最低6位:x = x & 077。以上用法都先要設(shè)計(jì)好一個(gè)常數(shù),該常數(shù)只有需要的位是1,不需要的位是0。用它與指定的位串信息按位與。 (2)按位或運(yùn)算符(|) 按位或運(yùn)算將兩個(gè)運(yùn)算分量的對(duì)應(yīng)位按位遵照以下規(guī)則進(jìn)行計(jì)算: 0 | 0 = 0, 0 | 1 = 1, 1 | 0 = 1, 1 | 1 = 1 即只要有1個(gè)是1的位,結(jié)果為1,否則為0。 例如,023 | 035 結(jié)果為037。 按位或運(yùn)算的典型用法是將一個(gè)位串信息的某幾位置成1。如將要獲得最右4為1,其他位與變量j的其他位相同,可用邏輯或運(yùn)算017|j。若要把這結(jié)果賦給變量j,可寫成: j = 017|j (3)按位異或運(yùn)算符(^) 按位異或運(yùn)算將兩個(gè)運(yùn)算分量的對(duì)應(yīng)位按位遵照以下規(guī)則進(jìn)行計(jì)算: 0 ^ 0 = 0, 0 ^ 1 = 1, 1 ^ 0 = 1, 1 ^ 1 = 0 即相應(yīng)位的值相同的,結(jié)果為 0,不相同的結(jié)果為 1。 例如,013^035結(jié)果為026。 異或運(yùn)算的意思是求兩個(gè)運(yùn)算分量相應(yīng)位值是否相異,相異的為1,相同的為0。按位異或運(yùn)算的典型用法是求一個(gè)位串信息的某幾位信息的反。如欲求整型變量j的最右4位信息的反,用邏輯異或運(yùn)算017^j,就能求得j最右4位的信息的反,即原來為1的位,結(jié)果是0,原來為0的位,結(jié)果是1。 (4)按位取反運(yùn)算符(~) 按位取反運(yùn)算是單目運(yùn)算,用來求一個(gè)位串信息按位的反,即哪些為0的位,結(jié)果是1,而哪些為1的位,結(jié)果是0。例如, ~7的結(jié)果為0xfff8。 取反運(yùn)算常用來生成與系統(tǒng)實(shí)現(xiàn)無關(guān)的常數(shù)。如要將變量x最低6位置成0,其余位不變,可用代碼x = x & ~077實(shí)現(xiàn)。以上代碼與整數(shù)x用2個(gè)字節(jié)還是用4個(gè)字節(jié)實(shí)現(xiàn)無關(guān)。 當(dāng)兩個(gè)長(zhǎng)度不同的數(shù)據(jù)進(jìn)行位運(yùn)算時(shí)(例如long型數(shù)據(jù)與int型數(shù)據(jù)),將兩個(gè)運(yùn)算分量的右端對(duì)齊進(jìn)行位運(yùn)算。如果短的數(shù)為正數(shù),高位用0補(bǔ)滿;如果短的數(shù)為負(fù)數(shù),高位用1補(bǔ)滿。如果短的為無符號(hào)整數(shù),則高位總是用0補(bǔ)滿。 位運(yùn)算用來對(duì)位串信息進(jìn)行運(yùn)算,得到位串信息結(jié)果。如以下代碼能取下整型變量k的位串信息的最右邊為1的信息位:((k-1)^k) & k。 移位運(yùn)算 移位運(yùn)算用來將整型或字符型數(shù)據(jù)作為二進(jìn)位信息串作整體移動(dòng)。有兩個(gè)運(yùn)算符: << (左移) 和 >> (右移) 移位運(yùn)算是雙目運(yùn)算,有兩個(gè)運(yùn)算分量,左分量為移位數(shù)據(jù)對(duì)象,右分量的值為移位位數(shù)。移位運(yùn)算將左運(yùn)算分量視作由二進(jìn)位組成的位串信息,對(duì)其作向左或向右移位,得到新的位串信息。 移位運(yùn)算符的優(yōu)先級(jí)低于算術(shù)運(yùn)算符,高于關(guān)系運(yùn)算符,它們的結(jié)合方向是自左至右。 (1)左移運(yùn)算符(<<) 左移運(yùn)算將一個(gè)位串信息向左移指定的位,右端空出的位用0補(bǔ)充。例如014<<2,結(jié)果為060,即48。 左移時(shí),空出的右端用0補(bǔ)充,左端移出的位的信息就被丟棄。在二進(jìn)制數(shù)運(yùn)算中,在信息沒有因移動(dòng)而丟失的情況下,每左移1位相當(dāng)于乘2。如4 << 2,結(jié)果為16。 (2)右移運(yùn)算符(>>) 右移運(yùn)算將一個(gè)位串信息向右移指定的位,右端移出的位的信息被丟棄。例如12>>2,結(jié)果為3。與左移相反,對(duì)于小整數(shù),每右移1位,相當(dāng)于除以2。在右移時(shí),需要注意符號(hào)位問題。對(duì)無符號(hào)數(shù)據(jù),右移時(shí),左端空出的位用0補(bǔ)充。對(duì)于帶符號(hào)的數(shù)據(jù),如果移位前符號(hào)位為0(正數(shù)),則左端也是用0補(bǔ)充;如果移位前符號(hào)位為1(負(fù)數(shù)),則左端用0或用1補(bǔ)充,取決于計(jì)算機(jī)系統(tǒng)。對(duì)于負(fù)數(shù)右移,稱用0 補(bǔ)充的系統(tǒng)為“邏輯右移”,用1補(bǔ)充的系統(tǒng)為“算術(shù)右移”。以下代碼能說明讀者上機(jī)的系統(tǒng)所采用的右移方法: printf("%d/n/n/n", -2>>4); 若輸出結(jié)果為-1,是采用算術(shù)右移;輸出結(jié)果為一個(gè)大整數(shù),則為邏輯右移。 移位運(yùn)算與位運(yùn)算結(jié)合能實(shí)現(xiàn)許多與位串運(yùn)算有關(guān)的復(fù)雜計(jì)算。設(shè)變量的位自右至左順序編號(hào),自0位至15位,有關(guān)指定位的表達(dá)式是不超過15的正整數(shù)。以下各代碼分別有它們右邊注釋所示的意義: ~(~0 << n) (x >> (1+p-n)) & ~(~0 << n) new |= ((old >> row) & 1) << (15 – k) s &= ~(1 << j) for(j = 0; ((1 << j) & s) == 0; j++) ; 本算法只采用移位、加減法、判斷和循環(huán)實(shí)現(xiàn),因?yàn)樗恍枰↑c(diǎn)運(yùn)算,也不需要乘除運(yùn)算,因此可以很方便地運(yùn)用到各種芯片上去。 我們先來看看10進(jìn)制下是如何手工計(jì)算開方的。 先看下面兩個(gè)算式, x = 10*p + q (1) 公式(1)左右平方之后得: x^2 = 100*p^2 + 20pq + q^2 (2) 現(xiàn)在假設(shè)我們知道x^2和p,希望求出q來,求出了q也就求出了x^2的開方x了。 我們把公式(2)改寫為如下格式: q = (x^2 - 100*p^2)/(20*p+q) (3) 這個(gè)算式左右都有q,因此無法直接計(jì)算出q來,因此手工的開方算法和手工除法算法一樣有一步需要猜值。 我們來一個(gè)手工計(jì)算的例子:計(jì)算1234567890的開方 首先我們把這個(gè)數(shù)兩位兩位一組分開,計(jì)算出最高位為3。也就是(3)中的p,最下面一行的334為余數(shù),也就是公式(3)中的(x^2 - 100*p^2)近似值 3 --------------- / 12 34 56 78 90 9 --------------- / 3 34 下面我們要找到一個(gè)0-9的數(shù)q使它最接近滿足公式(3)。我們先把p乘以20寫在334左邊: 3 q --------------- / 12 34 56 78 90 9 --------------- (20*3+q)*q / 3 34 我們看到q為5時(shí)(60+q)*q的值最接近334,而且不超過334。于是我們得到: 3 5 --------------- / 12 34 56 78 90 9 --------------- 65 / 3 34 3 25 --------------- 9 56 接下來就是重復(fù)上面的步驟了,這里就不再啰嗦了。 這個(gè)手工算法其實(shí)和10進(jìn)制關(guān)系不大,因此我們可以很容易的把它改為二進(jìn)制,改為二進(jìn)制之后,公式(3)就變成了: q = (x^2 - 4*p^2)/(4*p+q) (4) 我們來看一個(gè)例子,計(jì)算100(二進(jìn)制1100100)的開方: 1 0 1 0 ----------- / 1 10 01 00 1 ----------- 100 / 0 10 0 00 ----------- 1001 / 10 01 10 01 ----------- 0 00 這里每一步不再是把p乘以20了,而是把p乘以4,也就是把p右移兩位,而由于q的值只能為0或者1,所以我們只需要判斷余數(shù)(x^2 - 4*p^2)和(4*p+1)的大小關(guān)系,如果余數(shù)大于等于(4*p+q)那么該上一個(gè)1,否則該上一個(gè)0。 下面給出完成的C語言程序,其中root表示p,rem表示每步計(jì)算之后的余數(shù),divisor表示(4*p+1),通過a>>30取a的最高 2位,通過a<<=2將計(jì)算后的最高2位剔除。其中root的兩次<<1相當(dāng)于4*p。程序完全是按照手工計(jì)算改寫的,應(yīng)該不難理解。 unsigned short sqrt(unsigned long a){ unsigned long rem = 0; unsigned long root = 0; unsigned long divisor = 0; for(int i=0; i<16; ++i){ root <<= 1; rem = ((rem << 2) + (a >> 30)); a <<= 2; divisor = (root<<1) + 1; if(divisor <= rem){ rem -= divisor; root++; } } return (unsigned short)(root); } #define SetBit(x,y) x|=(1<<(y - 1)) //將X的第Y位置1 #define ClrBit(x,y) x&=~(1<<(y - 1)) //將X的第Y位清0
bool GetBit(int x, int y) //判斷某位是否為1 { x = x>>(y - 1); if((x&1) == 1) { return true; } else { return false; } } 使用32位整形標(biāo)示多個(gè)屬性,總共拆分了4個(gè)段,標(biāo)示不同的意義,如下,如果要靈活拆分,只要&上不同長(zhǎng)度的1再移位就可以了。 #define get_partionID(taskID) ((u8)((taskID&0xfe000000)>>25)) //26位到32位(共7位) #define get_blkID(taskID) ((u16)((taskID&0x01fff800)>>11)) //低12位到25位(共14位) #define get_recordID(taskID) ((u8)((taskID&0x000007E0)>>5)) //低6位到11位(共6位) #define get_tag(taskID) ((u8)((taskID&0x0000001f))) //取求低5位 #define get_RpID(taskID) 0
|