Objective-C中的位運算符用法

Objective-C語言中有各種各樣的運算符可處理數字中的特定位,如下表所示:

符號運算

& 按位與
| 按位或
^ 按位異或
~ 一次求反
<< 向左移位
>> 向右移位

表中列出的所有運算符,除一次求反運算符(~)外,都是二元運算符,因此需要兩個運算數。位運算符可處理任何類型的整型值,但不能處理浮點值。

1、按位運算符

對兩個值執行與運算時,會逐位比較兩個值的二進制表示。第一個值與第二個值對應位都為1時,在結果的對應位上就會得到1,其他的組合在結果中都得到0。如果b1和b2表示兩個運算數的對應位,那么下表(稱為真值表)就顯示了在b1和b2所有可能值下對b1和b2執行與操作的結果。

b1 b2 b1 & b2
——————————
0 0 0
0 1 0
1 0 0
1 1 1

例如,如果w1和w2都定義為short int , w1等于十六進制的15 , w2等于十六進制的0c,那么以下C語句會將值0x04指派給w3。

w3 = w1 & w2;

將w1、w2和w3都表示為二進制后可更清楚地看到此過程,假設所處理的short int大小為16位。

w1 0000 0000 0001 0101 0x15
w2 0000 0000 0000 1100 & 0x0c
————————————————————
w3 0000 0000 0000 0100 0x04

按位與運算經常用于屏蔽運算。就是說,這個運算符可輕易地將數據項的特定位設置為0。例如,語句

w3 = w1 & 3;

將w1與常量3按位與所得的值指派給w3。它的作用是將w3中的全部位(而非最右邊的兩位)設置為0,并保留w1中最左邊的兩位。

與Objective-C中使用的所有二元運算符相同,通過添加等號,二元位運算符可同樣用作賦值運算符。因此語句

word &= 15;

與下列語句

word = word & 15;

執行相同的功能。

此外,它還能將word的全部位設置為0,但最右邊的四位除外。

2、按位或運算符

在Objective-C中對兩個值執行按位或運算時,會逐位比較兩個值的二進制表示。此時,只要第一個值或者第二個值的相應位是1。那么結果的對應位就是1。按位或操作符的真值表如下所示。

b1 b2 b1 | b2
———————————
0 0 0
0 1 1
1 0 1
1 1 1

所以,如果w1是short int,等于十六進制的19, w2也是short int,等于十六進制的6a,那么對w1和vv2執行按位或會得到十六進制的7b,如下所示:

w1 0000 0000 0001 1001 0x19
w2 0000 0000 0110 1010 | 0x6a
————————————————————
0000 0000 0111 1011 0x7b

按位或操作通常就稱為按位OR,用于將某個詞的特定位設為1。例如,以下語句將w1最右邊的三位設為1,而不管這些位操作前的狀態是什么都是如此。

w1 = w1 | 07;

當然,可以在語句中使用特殊的斌值運算符,如下面的語句所示:

w1 |= 07;

我們在后面會提供一個程序例子,演示如何使用按位或運算符。

3、按位異或運算符

按位異或運算符,通常稱為XOR運算符,遵守以下規則:對干兩個運算數的相應位,如果任何一個位是1,但不是兩者全為1,那么結果的對應位將是1,否則是0。該運算符的真值表如

下所示:

b1 b2 b1 ^ b2
————————————
0 0 0
0 1 1
1 0 1
1 1 0

如果w1和w2分別等于十六進制的5e和d6,那么w1與w2執行異或運算后的結果是十六進制值e8,如下所示:

w1 0000 0000 0101 1110 0x5e
w2 0000 0000 1011 0110 ^ 0xd6
——————————————————————
0000 0000 1110 1000 0xe8

本文就先講到這里,對于Objective-C位運算符我們下一篇繼續探討,下次主要討論一下Objective-C位運算符中的一次求反、向左移位運算、向右移位運算,下回見。

1、一次求反運算

一次求反運算符是一元運算符,它的作用僅是對運算數的位“翻轉”。將運算數的每個是1的位翻轉為0,而將每個是0的位翻轉為1。此處提供真值表只是為了保持內容的完整性。

b1 ~b1
——————
0 1
1 0

如果w1是short int, 16位長,等于十六進制值a52f,那么對該值執行一次求反運算會得到十六進制值5ab0:

w1 1010 0101 0010 1111 0xa52f
~w1 0101 1010 1101 0000 0x5ab0

如果不知道運算中數值的準確位大小,那么一次求反運算符非常有用,使用它可讓程序不會依賴于整數數據類型的特定大小。例如,要將類型為int的w1的最低位設為0,可將一個所有位都是1、但最右邊的位是0的int值與w1進行與運算。所以像下面這樣的C語句在用32位表示整數的機器上可正常工作。

w1 &= 0xFFFFFFFE;

如果用

w1 &= ~1;

替換上面的語句,那么在任何機器上w1都會同正確的值進行與運算。

這是因為這條語句會對1求反,然后在左側會加入足夠的1,以滿足int的大小要求(在32位機器上,會在左側的31個位上加入1)。

現在,顯示一個實際的程序例子,說明各種位運算符的用途

// Bitwise operators illustrated
#import
intmain (intargc,char*argv[])
{
NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
unsignedintw1 = 0xA0A0A0A0, w2 = 0xFFFF0000,
w3 = 0x00007777;
NSLog (@”%x %x %x”, w1 & w2, w1 | w2, w1 ^ w2);
NSLog (@”%x %x %x”, ~w1, ~w2, ~w3);
NSLog (@”%x %x %x”, w1 ^ w1, w1 & ~w2, w1 | w2 | w3);
NSLog (@”%x %x”, w1 | w2 & w3, w1 | w2 & ~w3);
NSLog (@”%x %x”, (w1 & ~w2), (w1 | ~w2));
[pool drain];
return0;
}

結果輸出:

a0a00000 ffffa0a0 5f5fa0a0
5f5f5f5f ffff ffff8888
0 a0a0 fffff7f7
a0a0a0a0 ffffa0a0
ffffa0a0 a0a00000

對代碼中的每個運算都演算一遍,確定你理解了這些結果是如何得到的。

在第四個NSLog調用中,需要注意重要的一點,即按位與運算符的優先級要高于按位或運算符,因為這會實際影響表達式的最終結果值。

第五個NSLog調用展示了DeMorgan的規則:(a & ~b)等于a | b,(a | ~b)等于a & b。

2、向左移位運算符

對值執行向左移位運算時,按照字面的意思,值中包含的位將向左移動。與該操作關聯的是該值要移動的位置(或位)數目。超出數據項的高位的位將丟失,而從低位移入的值總為0。因此,如果w1等于3,那么表達式

w1 = w1 << 1;

可同樣表示成

w1 <<= 1;

結果就是3向左移一位,這樣產生的6將賦值給w1。

w1 ... 0000 0011 0x03

w1 << 1 ... 0000 0110 0x06

3、向右移位運算符

顧名思義,向右移位運算符(>>)把值的位向右移動。從值的低位移出的位將丟失。把無符號的值向右移位總是左側(就是高位)移人0。對于有符號值而言,左側移入1還是0依賴于被移動數字的符號,還取決于該操作在計算機上的實現方式。如果符號位是0(表示該值是正的),不管哪種機器都將移人0。然而,如果符號位是1,那么在一些計算機上將移人1,而其他計算機上則移入0。前一類型的運算符通常稱為算術右移,而后者通常稱為邏輯右移。

如果w1是unsigned int,用32位表示它并且它等于+六進制的F777EE22,那么使用語句

w1 >>= 1;

將w1右移一位后,w1等于十六進制的7BBBF711,如下所示:

w1 1111 0111 0111 0111 1110 1110 0010 0010 0xF777EE22
w1 >> 1 0111 1011 1011 1011 1111 0111 0001 0001 0x7BBBF711

如果將w1聲明為(有符號)的short int,在某些計算機上會得到相同的結果;而在其他計算機上,如果將該運算作為算術右移來執行,結果將會是FBBBF711。

應該注意到,如果試圖用大于或等于該數據項的位數將值向左或向右移位,那么該Objective-C語言并不會產生規定的結果。因此,例如計算機用32位表示整數,那么把一個整數向左或向右移動32位或更多位時,并不會在計算機上產生規定的結果。還注意到,如果使用負數對值移位時,結果將同樣是未定義的。

本文轉自:http://www.cnblogs.com/pengyingh/articles/2357033.html

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,936評論 6 535
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,744評論 3 421
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事?!?“怎么了?”我有些...
    開封第一講書人閱讀 176,879評論 0 381
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,181評論 1 315
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,935評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,325評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,384評論 3 443
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,534評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,084評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,892評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,067評論 1 371
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,623評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,322評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,735評論 0 27
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,990評論 1 289
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,800評論 3 395
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,084評論 2 375

推薦閱讀更多精彩內容