c#二進(jìn)制逆序方法詳解
原題
一個(gè)整數(shù),可以表示為二進(jìn)制的形式,請(qǐng)給出盡可能多的方法對(duì)二進(jìn)制進(jìn)行逆序操作。 例如:10000110 11011000的逆序?yàn)?00011011 01100001
分析
題目中說是一個(gè)整數(shù),對(duì)它的二進(jìn)制進(jìn)行逆序。并不是一個(gè)01字符串,或者01的數(shù)組。那么我們?cè)撊绾谓鉀Q這個(gè)問題呢?方法還是比較多的,有的中規(guī)中矩、有的非常巧妙。我們要掌握中規(guī)中規(guī)的方法,見識(shí)更多的巧妙的方法。慢慢的,能夠舉一反三,在遇到新的問題時(shí),能夠有靈思妙想。
最直接的方法
直接的方法,很容易想到:有如下代碼:
直接的方法,很容易想到:有如下代碼: int v = 111;
int r = v;
int s = 32;
for (; 0 != v; v >>= 1) {
r <<= 1;
r |= v & 1;
s--;
}
r <<= s;
System.out.println(r);
代碼比較好理解,取到v的最低位,作為r的最高位;v每取一次最低位,則右移一位;r每確定一位,則左移一位。同時(shí)記錄移動(dòng)了多少位,最終要補(bǔ)齊。
通過查表的方法
在遇到位操作的問題時(shí),往往題目中限定了總的位數(shù),比如這個(gè)題目,我們可以認(rèn)為32位。這就給我們帶來了一個(gè)以空間換時(shí)間的解決思路:查表法。位數(shù)是固定的,可以申請(qǐng)空間,存儲(chǔ)預(yù)先計(jì)算好的結(jié)果,在計(jì)算其他的結(jié)果的時(shí)候,則查表即可。
32位相對(duì)于查表來講,還是太大了。既然這樣縮小范圍,32個(gè)bit,也就是4個(gè)byte。每個(gè)byte 8bit,可以表示0-255的整數(shù)??梢酝ㄟ^申請(qǐng)256大小的數(shù)組,保存這256個(gè)整數(shù),二進(jìn)制逆序之后的整數(shù)。然后將一個(gè)32位的整數(shù),劃分為4個(gè)byte,每一個(gè)byte查表得到逆序的整數(shù):r1,r2,r3,r4。按照r4r3r2r1順序拼接二進(jìn)制得到的結(jié)果就是最終的答案。
這是一個(gè)思路,大家可以進(jìn)一步思考,嘗試。
巧妙的方法
我們這里主要分析這個(gè)巧妙的方法,核心思想是:分治法。即:
•逆序32位分解為兩個(gè)逆序16位的
•逆序16位分解為兩個(gè)逆序8位的
•逆序8位分解為兩個(gè)逆序4位的
•逆序4位分解為兩個(gè)逆序2位的
最后一個(gè)2位的逆序,直接交換即可。也就是分治遞歸的終止條件。但是,在上面的過程中,還沒有應(yīng)用到位操作的技巧。根據(jù)動(dòng)態(tài)規(guī)劃的思想,我們可以自底向上的解決這個(gè)問題:
•每2位為一組,進(jìn)行交換,完成2位逆序
•每4位為一組,前面2位與后面2位交換,完成4位逆序
•每8位為一組,前面4位和后面4為交換,完成8位的逆序
•每16位為一組,前面8位和后面8位交換,完成16位的逆序
2組16位的交換,完成32位的逆序
通過下面的例子,詳解上面的過程,我們以16位為例:10000110 11011000
1 | 0 | 0 | 0 | 0 | 1 | 1 | 0 | 1 | 1 | 0 | 1 | 1 | 0 | 0 | 0 |
0 | 1 | 0 | 0 | 1 | 0 | 0 | 1 | 1 | 1 | 1 | 0 | 0 | 1 | 0 | 0 |
0 | 0 | 0 | 1 | 0 | 1 | 1 | 0 | 1 | 0 | 1 | 1 | 0 | 0 | 0 | 1 |
0 | 1 | 1 | 0 | 0 | 0 | 0 | 1 | 0 | 0 | 0 | 1 | 1 | 0 | 1 | 1 |
0 | 0 | 0 | 1 | 1 | 0 | 1 | 1 | 0 | 1 | 1 | 0 | 0 | 0 | 0 | 1 |
經(jīng)過4步,逆序完成。推而廣之,總的時(shí)間復(fù)雜度為O(logn),n是二進(jìn)制的位數(shù)。這個(gè)方法可以推廣到任意位。
示例代碼如下:
int v =111;
v =((v >>1)&0x55555555)|((v &0x55555555)<<1);
v =((v >>2)&0x33333333)|((v &0x33333333)<<2);
v =((v >>4)&0x0F0F0F0F)|((v &0x0F0F0F0F)<<4);
v =((v >>8)&0x00FF00FF)|((v &0x00FF00FF)<<8);
v =( v >>16)|( v <<16);System.out.println(v);上面的思路理解了,代碼不難理解。例如第二行,前邊是取偶數(shù)位,后面是取奇數(shù)位,奇數(shù)位左移一位,偶數(shù)位右移一位,再取或,就是交換了奇數(shù)偶數(shù)位。也就是第一個(gè)步驟。
基于位運(yùn)算的一些巧妙的方法有很多。大家可以自行研究,后面會(huì)和大家分享更多的面試題目。
【分析完畢】
- C#創(chuàng)建安全的棧(Stack)存儲(chǔ)結(jié)構(gòu)
- C#使用Object類實(shí)現(xiàn)棧的方法詳解
- C#數(shù)據(jù)結(jié)構(gòu)之堆棧(Stack)實(shí)例詳解
- 一看就懂:圖解C#中的值類型、引用類型、棧、堆、ref、out
- C#使用foreach語句遍歷堆棧(Stack)的方法
- 淺談C#中堆和棧的區(qū)別(附上圖解)
- c#棧變化規(guī)則圖解示例(棧的生長(zhǎng)與消亡)
- 解析c#在未出現(xiàn)異常情況下查看當(dāng)前調(diào)用堆棧的解決方法
- C#棧和堆的區(qū)別淺談
- C#數(shù)據(jù)結(jié)構(gòu)與算法揭秘五 棧和隊(duì)列
- C#遞歸實(shí)現(xiàn)將一整數(shù)逆序后放入一數(shù)組中
- C#實(shí)現(xiàn)用棧求逆序的方法示例
相關(guān)文章
C#?WPF?ListBox?動(dòng)態(tài)顯示圖片功能
這篇文章主要介紹了C#?WPF?ListBox?動(dòng)態(tài)顯示圖片,處理過程分為前臺(tái)代碼和后臺(tái)代碼,本文結(jié)合實(shí)例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下2023-08-08C#將Word或Excel文檔轉(zhuǎn)換為Html文件
這篇文章介紹了C#將Word或Excel文檔轉(zhuǎn)換為Html文件的方法,文中通過示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-06-06C#設(shè)置軟件開機(jī)自動(dòng)運(yùn)行的方法(修改注冊(cè)表)
這篇文章主要介紹了C#設(shè)置軟件開機(jī)自動(dòng)運(yùn)行的方法,通過簡(jiǎn)單修改注冊(cè)表開機(jī)啟動(dòng)項(xiàng)實(shí)現(xiàn)軟件的開機(jī)啟動(dòng)功能,非常簡(jiǎn)單實(shí)用,需要的朋友可以參考下2016-06-06Unity實(shí)現(xiàn)簡(jiǎn)單的虛擬搖桿
這篇文章主要為大家詳細(xì)介紹了Unity實(shí)現(xiàn)簡(jiǎn)單的虛擬搖桿,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-04-04C#實(shí)現(xiàn)公式計(jì)算驗(yàn)證碼的示例詳解
現(xiàn)在很多的平臺(tái)已經(jīng)不使用普通的數(shù)字、字母等驗(yàn)證碼了,取而代之的是拼圖類、選圖類、旋轉(zhuǎn)類或者計(jì)算類的驗(yàn)證碼。本文將利用C#實(shí)現(xiàn)一個(gè)公式計(jì)算驗(yàn)證碼,感興趣的可以了解一下2022-10-10C#數(shù)據(jù)結(jié)構(gòu)之單鏈表(LinkList)實(shí)例詳解
這篇文章主要介紹了C#數(shù)據(jù)結(jié)構(gòu)之單鏈表(LinkList)實(shí)現(xiàn)方法,結(jié)合實(shí)例形式較為詳細(xì)的分析了單鏈表的原理、定義與C#具體實(shí)現(xiàn)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-11-11C#數(shù)值轉(zhuǎn)換-隱式數(shù)值轉(zhuǎn)換表參考
隱式轉(zhuǎn)換就是直接使用,比如可以把一個(gè) byte 類型直接用在 int 上2013-04-04