色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

匯總幾個在算法題以及工程開發中常用的位運算技巧

算法與數據結構 ? 來源:labuladong ? 2023-03-13 09:16 ? 次閱讀

位操作(Bit Manipulation)可以有很多技巧,有一個叫做 Bit Twiddling Hacks 的網站收集了幾乎所有位操作的黑科技玩法

但是這些技巧大部分都過于晦澀,我覺得可以作為字典查閱,沒必要逐條深究。但我認為那些有趣的、有用的位運算技巧,是我們每個人需要掌握的。

所以本文由淺入深,先展示幾個有趣(但沒卵用)的位運算技巧,然后再匯總幾個在算法題以及工程開發中常用的位運算技巧。

幾個有趣的位操作

利用或操作`|`和空格將英文字符轉換為小寫

('a'|'')='a'
('A'|'')='a'

利用與操作`&`和下劃線將英文字符轉換為大寫

('b'&'_')='B'
('B'&'_')='B'

利用異或操作`^`和空格進行英文字符大小寫互換

('d'^'')='D'
('D'^'')='d'

以上操作能夠產生奇特效果的原因在于 ASCII 編碼。ASCII 字符其實就是數字,恰巧空格和下劃線對應的數字通過位運算就能改變大小寫。有興趣的讀者可以查 ASCII 碼表自己算算,本文就不展開講了。

不用臨時變量交換兩個數

inta=1,b=2;
a^=b;
b^=a;
a^=b;
//現在a=2,b=1

加一

intn=1;
n=-~n;
//現在n=2

減一

intn=2;
n=~-n;
//現在n=1

判斷兩個數是否異號

intx=-1,y=2;
booleanf=((x^y)

如果說前 6 個技巧的用處不大,這第 7 個技巧還是比較實用的,利用的是補碼編碼的符號位。整數編碼最高位是符號位,負數的符號位是 1,非負數的符號位是 0,再借助異或的特性,可以判斷出兩個數字是否異號。

當然,如果不用位運算來判斷是否異號,需要使用 if else 分支,還挺麻煩的。你可能想利用乘積來判斷兩個數是否異號,但是這種處理方式容易造成整型溢出,從而出現錯誤。

index & (arr.length - 1) 的運用

我在單調棧解題套路中介紹過環形數組,其實就是利用求模(余數)的方式讓數組看起來頭尾相接形成一個環形,永遠都走不完:

int[]arr={1,2,3,4};
intindex=0;
while(true){
//在環形數組中轉圈
print(arr[index%arr.length]);
index++;
}
//輸出:1,2,3,4,1,2,3,4,1,2,3,4...

但模運算%對計算機來說其實是一個比較昂貴的操作,所以我們可以用&運算來求余數:

int[]arr={1,2,3,4};
intindex=0;
while(true){
//在環形數組中轉圈
print(arr[index&(arr.length-1)]);
index++;
}
//輸出:1,2,3,4,1,2,3,4,1,2,3,4...

簡單說,& (arr.length - 1)這個位運算能夠替代% arr.length的模運算,性能會更好一些。

那問題來了,現在是不斷地index++,你做到了循環遍歷。但如果不斷地index--,還能做到環形數組的效果嗎?

答案是,如果你使用%求模的方式,那么當index小于 0 之后求模的結果也會出現負數,你需要特殊處理。但通過&與運算的方式,index不會出現負數,依然可以正常工作:

int[]arr={1,2,3,4};
intindex=0;
while(true){
//在環形數組中轉圈
print(arr[index&(arr.length-1)]);
index--;
}
//輸出:1,4,3,2,1,4,3,2,1,4,3,2,1...

我們自己寫代碼一般用不到這個技巧,但在學習一些其他代碼庫時可能會經常看到,這里留個印象,到時候就不會懵逼了。

n & (n-1) 的運用

n & (n-1)這個操作在算法中比較常見,作用是消除數字n的二進制表示中的最后一個 1

看個圖就很容易理解了:

26848936-c01d-11ed-bfe3-dac502259ad0.png

其核心邏輯就是,n - 1一定可以消除最后一個 1,同時把其后的 0 都變成 1,這樣再和n做一次&運算,就可以僅僅把最后一個 1 變成 0 了。

1、計算漢明權重(Hamming Weight)

這是力扣第 191 題「位 1 的個數」:

2697b114-c01d-11ed-bfe3-dac502259ad0.png

就是讓你返回n的二進制表示中有幾個 1。因為n & (n - 1)可以消除最后一個 1,所以可以用一個循環不停地消除 1 同時計數,直到n變成 0 為止。

inthammingWeight(intn){
intres=0;
while(n!=0){
n=n&(n-1);
res++;
}
returnres;
}

2、判斷一個數是不是 2 的指數

力扣第 231 題「2 的冪」就是這個問題。

一個數如果是 2 的指數,那么它的二進制表示一定只含有一個 1:

2^0=1=0b0001
2^1=2=0b0010
2^2=4=0b0100

如果使用n & (n-1)的技巧就很簡單了(注意運算符優先級,括號不可以省略):

booleanisPowerOfTwo(intn){
if(n<=?0)?return?false;
????return?(n?&?(n?-?1))?==?0;
}

a ^ a = 0 的運用

異或運算的性質是需要我們牢記的:

一個數和它本身做異或運算結果為 0,即a ^ a = 0;一個數和 0 做異或運算的結果為它本身,即a ^ 0 = a。

1、查找只出現一次的元素

這是力扣第 136 題「只出現一次的數字」:

26af6e26-c01d-11ed-bfe3-dac502259ad0.png

對于這道題目,我們只要把所有數字進行異或,成對兒的數字就會變成 0,落單的數字和 0 做異或還是它本身,所以最后異或的結果就是只出現一次的元素:

intsingleNumber(int[]nums){
intres=0;
for(intn:nums){
res^=n;
}
returnres;
}

2、尋找缺失的元素

這是力扣第 268 題「丟失的數字」:

26cc5d60-c01d-11ed-bfe3-dac502259ad0.png

給一個長度為n的數組,其索引應該在[0,n),但是現在你要裝進去n + 1個元素[0,n],那么肯定有一個元素裝不下嘛,請你找出這個缺失的元素。

這道題不難的,我們應該很容易想到,把這個數組排個序,然后遍歷一遍,不就很容易找到缺失的那個元素了嗎?

或者說,借助數據結構的特性,用一個 HashSet 把數組里出現的數字都儲存下來,再遍歷[0,n]之間的數字,去 HashSet 中查詢,也可以很容易查出那個缺失的元素。

排序解法的時間復雜度是 O(NlogN),HashSet 的解法時間復雜度是 O(N),但是還需要 O(N) 的空間復雜度存儲 HashSet。

這個問題其實還有一個特別簡單的解法:等差數列求和公式。

題目的意思可以這樣理解:現在有個等差數列0, 1, 2,..., n,其中少了某一個數字,請你把它找出來。那這個數字不就是sum(0,1,..n) - sum(nums)嘛?

intmissingNumber(int[]nums){
intn=nums.length;
//雖然題目給的數據范圍不大,但嚴謹起見,用long類型防止整型溢出
//求和公式:(首項+末項)*項數/ 2
longexpect=(0+n)*(n+1)/2;
longsum=0;
for(intx:nums){
sum+=x;
}
return(int)(expect-sum);
}

不過,本文的主題是位運算,我們來講講如何利用位運算技巧來解決這道題。

再回顧一下異或運算的性質:一個數和它本身做異或運算結果為 0,一個數和 0 做異或運算還是它本身。

而且異或運算滿足交換律和結合律,也就是說:

2^3^2=3^(2^2)=3^0=3

而這道題索就可以通過這些性質巧妙算出缺失的那個元素,比如說nums = [0,3,1,4]:

26e32ce8-c01d-11ed-bfe3-dac502259ad0.jpg

為了容易理解,我們假設先把索引補一位,然后讓每個元素和自己相等的索引相對應:

26f2ff92-c01d-11ed-bfe3-dac502259ad0.jpg

這樣做了之后,就可以發現除了缺失元素之外,所有的索引和元素都組成一對兒了,現在如果把這個落單的索引 2 找出來,也就找到了缺失的那個元素。

如何找這個落單的數字呢,只要把所有的元素和索引做異或運算,成對兒的數字都會消為 0,只有這個落單的元素會剩下,也就達到了我們的目的:

intmissingNumber(int[]nums){
intn=nums.length;
intres=0;
//先和新補的索引異或一下
res^=n;
//和其他的元素、索引做異或
for(inti=0;i
270699d0-c01d-11ed-bfe3-dac502259ad0.jpg

由于異或運算滿足交換律和結合律,所以總是能把成對兒的數字消去,留下缺失的那個元素。

到這里,常見的位運算差不多都講完了。這些技巧就是會者不難難者不會,也不需要死記硬背,只要有個印象就完全夠用了。





審核編輯:劉清

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 計算機
    +關注

    關注

    19

    文章

    7508

    瀏覽量

    88070
  • ASCII
    +關注

    關注

    5

    文章

    172

    瀏覽量

    35121
  • Hash算法
    +關注

    關注

    0

    文章

    43

    瀏覽量

    7383

原文標題:必知必會位運算技巧手冊

文章出處:【微信號:TheAlgorithm,微信公眾號:算法與數據結構】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    【「從算法到電路—數字芯片算法的電路實現」閱讀體驗】+內容簡介

    。本書力求從算法、芯片設計、軟件開發等多個角度解讀基礎算法電路的設計,涵蓋了溢出保護、有符號運算、浮點運算
    發表于 11-21 17:14

    運算

    運算為了節省內存空間,系統軟件中常將多個標志狀態簡單地組合在一起,存儲到一個字節(或字)中。C語言是為研制系統軟件而設計的,所以她提供了實現將標志狀態從標志字節中分離出來的
    發表于 03-10 15:07

    嵌入式開發中常用的總線與接口匯總

    盤點嵌入式開發中常用的總線與接口
    發表于 02-01 07:25

    Matlab編程中常用的優化技巧

    用過Matlab的同學應該都知道,Matlab的慢是出了名的,但是再慢也有優化的方式,下面我們給出幾個Matlab編程中常用的優化技巧。??講優化方法之前,首先要說的就是Matlab中用tic
    發表于 02-19 06:40

    算法類型在運動控制中常用的有哪些

    算法類型在運動控制中常用的加減速控制算法有指數、直線、S型曲線和三角函數加減速控制算法。PS:S型曲線加減速關注度指數,近年在上升。沖擊類型和加加速度解釋剛性沖擊:速度發生突變,加速度
    發表于 09-03 08:57

    介紹開發ESP8266開發中常見的一些問題

    ESP8266 wifi模塊開發匯總 ESP8266 wifi模塊開發匯總本文檔主要介紹開發
    發表于 11-10 07:31

    嵌入式開發過程中常用的庫函數有哪些

    嵌入式開發過程中常用的庫函數有哪些?有何優勢?
    發表于 02-25 07:07

    c語言常用算法

    非常實用的《c語言常用算法程序集》針對工程中常用的行之有效的算法而編寫,其主要內容包括多項式的計算、復數
    發表于 04-11 16:41

    單片機常用PID濾波算法資料匯總

    單片機常用PID濾波算法資料匯總
    發表于 05-21 11:45 ?26次下載

    幾個Matlab編程中常用的優化技巧

    用過Matlab的同學應該都知道,Matlab的慢是出了名的,但是再慢也有優化的方式,下面我們給出幾個Matlab編程中常用的優化技巧。
    的頭像 發表于 02-08 15:18 ?3725次閱讀

    算法類型以及準備策略

    今天就和大家聊聊大公司的面試環節經常涉及的算法類型以及準備策略。 問題難度首先大家比較關心的就是面試時候出現的算法的難度,從我的個人經驗
    的頭像 發表于 09-02 10:50 ?1491次閱讀

    PCB中常用的快捷鍵匯總

    PCB中常用的快捷鍵匯總
    發表于 09-28 10:12 ?40次下載

    Vivado中常用TCL命令匯總

    Vivado是Xilinx推出的可編程邏輯設備(FPGA)軟件開發工具套件,提供了許多TCL命令來簡化流程和自動化開發。本文將介紹Vivado中常用的TCL命令,并對其進行詳細說明,
    的頭像 發表于 04-13 10:20 ?3775次閱讀

    STM32開發中的運算以及帶操作

    STM32開發中的運算以及帶操作? 運算是計算
    的頭像 發表于 02-02 14:38 ?1697次閱讀

    分享幾個嵌入式中常用的GUI

    交互,完成各種操作,可提高工作效率以及用戶體驗。接下來看一下我們開發中常用的GUI框架有哪些吧~二、開源輕量級顯示框架LVGLLVGL(LightandVersat
    的頭像 發表于 04-06 08:09 ?1626次閱讀
    分享<b class='flag-5'>幾個</b>嵌入式<b class='flag-5'>中常用</b>的GUI
    主站蜘蛛池模板: 久久电影院久久国产| 美国caopo超碰在线视频| 久久精品美女久久| 无码乱人伦一区二区亚洲一| 成年人深夜福利| 欧美2019高清hd巨大| 91九色麻豆| 免费国产成人手机在线观看| 伊人久久久久久久久久| 精品午夜中文字幕熟女人妻在线| 亚洲国产AV无码综合在线| 国产午夜精品不卡观看| 亚洲AV精品无码成人| 果冻传媒视频在线观看完整版免费 | 日本wwwxx| 国产AV精品白浆一区二| 桃花免费高清在线观看| 国产精品99久久久久久AV蜜臀| 特级黑人三人共一女| 国产精品一区二区在线观看| 午夜小视频免费观看| 精品国产午夜福利在线观看蜜月| 亚洲天堂999| 快播可乐网| 爱看吧孕妇网| 四虎成人影院| 精品国产九九| 91麻豆久久| 人善交XUANWEN200喷水| 国产VA精品午夜福利视频| 性欧美FREE少妇XXX| 久久精品电影院| 99久久免费看少妇高潮A片| 日韩精品无码久久一区二区三| 国产精品99精品无码视亚| 亚洲欧美中文字幕网站大全| 老妇高潮潮喷到猛进猛出| www黄色com| 亚洲精品成人a| 美女脱内衣裸身尿口露出来| 高清视频在线观看SEYEYE|