主頁 > 後端開發 > 經典面試題決議:這道 C 編程面試題居然有如此多的解法!

經典面試題決議:這道 C 編程面試題居然有如此多的解法!

2020-11-03 05:18:35 後端開發

問題描述

任意給定一個32位無符號整數n,求n的二進制表示中1的個數,比如n = 5(0101)時,回傳2,n = 15(1111)時,回傳4

這也是一道比較經典的題目了,相信不少人面試的時候可能遇到過這道題吧,下面介紹了幾種方法來實作這道題,相信很多人可能見過下面的演算法,但我相信很少有人見到本文中所有的演算法,如果您上頭上有更好的演算法,或者本文沒有提到的演算法,請不要吝惜您的代碼,分享的時候,也是學習和交流的時候,

普通法

我總是習慣叫普通法,因為我實在找不到一個合適的名字來描述它,其實就是最簡單的方法,有點程式基礎的人都能想得到,那就是移位+計數,很簡單,不多說了,直接上代碼,這種方法的運算次數與輸入n最高位1的位置有關,最多回圈32次,

int BitCount(unsigned int n)
{
    unsigned int c =0 ; // 計數器
    while (n >0)
    {
        if((n &1) ==1) // 當前位是1
            ++c ; // 計數器加1
        n >>=1 ; // 移位
    }
    return c ;
}

一個更精簡的版本如下

int BitCount1(unsigned int n)
{
    unsigned int c =0 ; // 計數器
    for (c =0; n; n >>=1) // 回圈移位
        c += n &1 ; // 如果當前位是1,則計數器加1
    return c ;
}

問:如果輸入引數是int,這種方法還能奏效嗎?

如何修改?看看下面的方法:

int BitCount(int n)
{
    int num = 0;
    unsigned int flag = 1;
    while(0 != flag)
    {
        if(n & flag)
            num++;
        flag = flag << 1;
    }
    return num;
}

 

C/C++的學習裙【七一二 二八四 七零五 】,無論你是小白還是進階者,是想轉行還是想入行都可以來了解一起進步一起學習!裙內有開發工具,很多干貨和技術資料分享!

快速法

這種方法速度比較快,其運算次數與輸入n的大小無關,只與n中1的個數有關,如果n的二進制表示中有k個1,那么這個方法只需要回圈k次即可,其原理是不斷清除n的二進制表示中最右邊的1,同時累加計數器,直至n為0,代碼如下

int BitCount2(unsigned int n)
{
    unsigned int c =0 ;
    for (c =0; n; ++c)
    {
        n &= (n -1) ; // 清除最低位的1
    }
    return c ;
}

為什么n &= (n – 1)能清除最右邊的1呢?因為從二進制的角度講,n相當于在n - 1的最低位加上1,舉個例子,8(1000)= 7(0111)+ 1(0001),所以8 & 7 = (1000)&(0111)= 0(0000),清除了8最右邊的1(其實就是最高位的1,因為8的二進制中只有一個1),再比如7(0111)= 6(0110)+ 1(0001),所以7 & 6 = (0111)&(0110)= 6(0110),清除了7的二進制表示中最右邊的1(也就是最低位的1),

查表法

動態建表

由于表示在程式運行時動態創建的,所以速度上肯定會慢一些,把這個版本放在這里,有兩個原因

  1. 介紹填表的方法,因為這個方法的確很巧妙,

  2. 型別轉換,這里不能使用傳統的強制轉換,而是先取地址再轉換成對應的指標型別,也是常用的型別轉換方法,

int BitCount3(unsigned int n) 

    // 建表
    unsigned char BitsSetTable256[256] = {0} ; 

    // 初始化表 
    for (int i =0; i <256; i++) 
    { 
        BitsSetTable256[i] = (i &1) + BitsSetTable256[i /2]; 
    } 

    unsigned int c =0 ; 

    // 查表
    unsigned char* p = (unsigned char*) &n ; 

    c = BitsSetTable256[p[0]] + 
        BitsSetTable256[p[1]] + 
        BitsSetTable256[p[2]] + 
        BitsSetTable256[p[3]]; 

    return c ; 
}

先說一下填表的原理,根據奇偶性來分析,對于任意一個正整數n

1.如果它是偶數,那么n的二進制中1的個數與n/2中1的個數是相同的,比如4和2的二進制中都有一個1,6和3的二進制中都有兩個1,為啥?因為n是由n/2左移一位而來,而移位并不會增加1的個數,

2.如果n是奇數,那么n的二進制中1的個數是n/2中1的個數+1,比如7的二進制中有三個1,7/2 = 3的二進制中有兩個1,為啥?因為當n是奇數時,n相當于n/2左移一位再加1,

再說一下查表的原理

對于任意一個32位無符號整數,將其分割為4部分,每部分8bit,對于這四個部分分別求出1的個數,再累加起來即可,而8bit對應2^8 = 256種01組合方式,這也是為什么表的大小為256的原因,

注意型別轉換的時候,先取到n的地址,然后轉換為unsigned char*,這樣一個unsigned int(4 bytes)對應四個unsigned char(1 bytes),分別取出來計算即可,舉個例子吧,以87654321(十六進制)為例,先寫成二進制形式-8bit一組,共四組,以不同顏色區分,這四組中1的個數分別為4,4,3,2,所以一共是13個1,如下面所示,

10000111 01100101 01000011 00100001 = 4 + 4 + 3 + 2 = 13

靜態表-4bit

原理和8-bit表相同,詳見8-bit表的解釋

int BitCount4(unsigned int n)
{
    unsigned int table[16] = 
    {
        0, 1, 1, 2, 
        1, 2, 2, 3, 
        1, 2, 2, 3, 
        2, 3, 3, 4
    } ;

    unsigned int count =0 ;
    while (n)
    {
        count += table[n &0xf] ;
        n >>=4 ;
    }
    return count ;
}

靜態表-8bit

首先構造一個包含256個元素的表table,table[i]即i中1的個數,這里的i是[0-255]之間任意一個值,然后對于任意一個32bit無符號整數n,我們將其拆分成四個8bit,然后分別求出每個8bit中1的個數,再累加求和即可,這里用移位的方法,每次右移8位,并與0xff相與,取得最低位的8bit,累加后繼續移位,如此往復,直到n為0,所以對于任意一個32位整數,需要查表4次,以十進制數2882400018為例,其對應的二進制數為10101011110011011110111100010010,對應的四次查表程序如下:紅色表示當前8bit,綠色表示右移后高位補零,

第一次(n & 0xff)          10101011110011011110111100010010

第二次((n >> 8) & 0xff)  00000000101010111100110111101111

第三次((n >> 16) & 0xff)00000000000000001010101111001101

第四次((n >> 24) & 0xff)00000000000000000000000010101011

int BitCount7(unsigned int n)

    unsigned int table[256] = 
    { 
        0, 1, 1, 2, 1, 2, 2, 3, 1, 2, 2, 3, 2, 3, 3, 4, 
        1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 
        1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 
        2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 
        1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 
        2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 
        2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 
        3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7, 
        1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 
        2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 
        2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 
        3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7, 
        2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 
        3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7, 
        3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7, 
        4, 5, 5, 6, 5, 6, 6, 7, 5, 6, 6, 7, 6, 7, 7, 8, 
    }; 

    return table[n &0xff] +
        table[(n >>8) &0xff] +
        table[(n >>16) &0xff] +
        table[(n >>24) &0xff] ;
}

當然也可以搞一個16bit的表,或者更極端一點32bit的表,速度將會更快,

平行演算法

網上都這么叫,我也這么叫吧,不過話說回來,的確有平行的意味在里面,先看代碼,稍后解釋

int BitCount4(unsigned int n) 

    n = (n &0x55555555) + ((n >>1) &0x55555555) ; 
    n = (n &0x33333333) + ((n >>2) &0x33333333) ; 
    n = (n &0x0f0f0f0f) + ((n >>4) &0x0f0f0f0f) ; 
    n = (n &0x00ff00ff) + ((n >>8) &0x00ff00ff) ; 
    n = (n &0x0000ffff) + ((n >>16) &0x0000ffff) ; 

    return n ; 
}

速度不一定最快,但是想法絕對巧妙,說一下其中奧妙,其實很簡單,先將n寫成二進制形式,然后相鄰位相加,重復這個程序,直到只剩下一位,

以217(11011001)為例,有圖有真相,下面的圖足以說明一切了,217的二進制表示中有5個1

完美法

int BitCount5(unsigned int n) 
{
    unsigned int tmp = n - ((n >>1) &033333333333) - ((n >>2) &011111111111);
    return ((tmp + (tmp >>3)) &030707070707) %63;
}

最喜歡這個,代碼太簡潔啦,只是有個取模運算,可能速度上慢一些,區區兩行代碼,就能計算出1的個數,到底有何奧妙呢?為了解釋的清楚一點,我盡量多說幾句,

第一行代碼的作用

先說明一點,以0開頭的是8進制數,以0x開頭的是十六進制數,上面代碼中使用了三個8進制數,

將n的二進制表示寫出來,然后每3bit分成一組,求出每一組中1的個數,再表示成二進制的形式,比如n = 50,其二進制表示為110010,分組后是110和010,這兩組中1的個數本別是2和3,2對應010,3對應011,所以第一行代碼結束后,tmp = 010011,具體是怎么實作的呢?由于每組3bit,所以這3bit對應的十進制數都能表示為2^2 * a + 2^1 * b + c的形式,也就是4a + 2b + c的形式,這里a,b,c的值為0或1,如果為0表示對應的二進制位上是0,如果為1表示對應的二進制位上是1,所以a + b + c的值也就是4a + 2b + c的二進制數中1的個數了,舉個例子,十進制數6(0110)= 4 * 1 + 2 * 1 + 0,這里a = 1, b = 1, c = 0, a + b + c = 2,所以6的二進制表示中有兩個1,現在的問題是,如何得到a + b + c呢?注意位運算中,右移一位相當于除2,就利用這個性質!

4a + 2b + c 右移一位等于2a + b

4a + 2b + c 右移量位等于a

然后做減法

4a + 2b + c –(2a + b) – a = a + b + c,這就是第一行代碼所作的事,明白了吧,

 

第二行代碼的作用

在第一行的基礎上,將tmp中相鄰的兩組中1的個數累加,由于累加到程序中有些組被重復加了一次,所以要舍棄這些多加的部分,這就是&030707070707的作用,又由于最終結果可能大于63,所以要取模,

需要注意的是,經過第一行代碼后,從右側起,每相鄰的3bit只有四種可能,即000, 001, 010, 011,為啥呢?因為每3bit中1的個數最多為3,所以下面的加法中不存在進位的問題,因為3 + 3 = 6,不足8,不會產生進位,

tmp + (tmp >> 3)-這句就是是相鄰組相加,注意會產生重復相加的部分,比如tmp = 659 = 001 010 010 011時,tmp >> 3 = 000 001 010 010,相加得

001 010 010 011

000 001 010 010


001 011 100 101

011 + 101 = 3 + 5 = 8,(感謝網友Di哈指正,)注意,659只是個中間變數,這個結果不代表659這個數的二進制形式中有8個1,

注意我們想要的只是第二組和最后一組(綠色部分),而第一組和第三組(紅色部分)屬于重復相加的部分,要消除掉,這就是&030707070707所完成的任務(每隔三位洗掉三位),最后為什么還要%63呢?因為上面相當于每次計算相連的6bit中1的個數,最多是111111 = 77(八進制)= 63(十進制),所以最后要對63取模,

位標志法

感謝網友 @gussing提供

struct _byte 

    unsigned a:1; 
    unsigned b:1; 
    unsigned c:1; 
    unsigned d:1; 
    unsigned e:1; 
    unsigned f:1; 
    unsigned g:1; 
    unsigned h:1; 
}; 

long get_bit_count( unsigned char b ) 
{
    struct _byte *by = (struct _byte*)&b; 
    return (by->a+by->b+by->c+by->d+by->e+by->f+by->g+by->h); 
}

來源:https://www.cnblogs.com/graphics/archive/2010/06/21/1752421.html

 

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/200480.html

標籤:C

上一篇:請問大神到地使出了什么問題,為什么c,b輸不上。

下一篇:【Flutter 混合開發】添加 Flutter 到 iOS

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • Rust中的智能指標:Box<T> Rc<T> Arc<T> Cell<T> RefCell<T> Weak

    Rust中的智能指標是什么 智能指標(smart pointers)是一類資料結構,是擁有資料所有權和額外功能的指標。是指標的進一步發展 指標(pointer)是一個包含記憶體地址的變數的通用概念。這個地址參考,或 ” 指向”(points at)一些其 他資料 。參考以 & 符號為標志并借用了他們所 ......

    uj5u.com 2023-04-20 07:24:10 more
  • Java的值傳遞和參考傳遞

    值傳遞不會改變本身,參考傳遞(如果傳遞的值需要實體化到堆里)如果發生修改了會改變本身。 1.基本資料型別都是值傳遞 package com.example.basic; public class Test { public static void main(String[] args) { int ......

    uj5u.com 2023-04-20 07:24:04 more
  • [2]SpinalHDL教程——Scala簡單入門

    第一個 Scala 程式 shell里面輸入 $ scala scala> 1 + 1 res0: Int = 2 scala> println("Hello World!") Hello World! 檔案形式 object HelloWorld { /* 這是我的第一個 Scala 程式 * 以 ......

    uj5u.com 2023-04-20 07:23:58 more
  • 理解函式指標和回呼函式

    理解 函式指標 指向函式的指標。比如: 理解函式指標的偽代碼 void (*p)(int type, char *data); // 定義一個函式指標p void func(int type, char *data); // 宣告一個函式func p = func; // 將指標p指向函式func ......

    uj5u.com 2023-04-20 07:23:52 more
  • Django筆記二十五之資料庫函式之日期函式

    本文首發于公眾號:Hunter后端 原文鏈接:Django筆記二十五之資料庫函式之日期函式 日期函式主要介紹兩個大類,Extract() 和 Trunc() Extract() 函式作用是提取日期,比如我們可以提取一個日期欄位的年份,月份,日等資料 Trunc() 的作用則是截取,比如 2022-0 ......

    uj5u.com 2023-04-20 07:23:45 more
  • 一天吃透JVM面試八股文

    什么是JVM? JVM,全稱Java Virtual Machine(Java虛擬機),是通過在實際的計算機上仿真模擬各種計算機功能來實作的。由一套位元組碼指令集、一組暫存器、一個堆疊、一個垃圾回收堆和一個存盤方法域等組成。JVM屏蔽了與作業系統平臺相關的資訊,使得Java程式只需要生成在Java虛擬機 ......

    uj5u.com 2023-04-20 07:23:31 more
  • 使用Java接入小程式訂閱訊息!

    更新完微信服務號的模板訊息之后,我又趕緊把微信小程式的訂閱訊息給實作了!之前我一直以為微信小程式也是要企業才能申請,沒想到小程式個人就能申請。 訊息推送平臺🔥推送下發【郵件】【短信】【微信服務號】【微信小程式】【企業微信】【釘釘】等訊息型別。 https://gitee.com/zhongfuch ......

    uj5u.com 2023-04-20 07:22:59 more
  • java -- 緩沖流、轉換流、序列化流

    緩沖流 緩沖流, 也叫高效流, 按照資料型別分類: 位元組緩沖流:BufferedInputStream,BufferedOutputStream 字符緩沖流:BufferedReader,BufferedWriter 緩沖流的基本原理,是在創建流物件時,會創建一個內置的默認大小的緩沖區陣列,通過緩沖 ......

    uj5u.com 2023-04-20 07:22:49 more
  • Java-SpringBoot-Range請求頭設定實作視頻分段傳輸

    老實說,人太懶了,現在基本都不喜歡寫筆記了,但是網上有關Range請求頭的文章都太水了 下面是抄的一段StackOverflow的代碼...自己大修改過的,寫的注釋挺全的,應該直接看得懂,就不解釋了 寫的不好...只是希望能給視頻網站開發的新手一點點幫助吧. 業務場景:視頻分段傳輸、視頻多段傳輸(理 ......

    uj5u.com 2023-04-20 07:22:42 more
  • Windows 10開發教程_編程入門自學教程_菜鳥教程-免費教程分享

    教程簡介 Windows 10開發入門教程 - 從簡單的步驟了解Windows 10開發,從基本到高級概念,包括簡介,UWP,第一個應用程式,商店,XAML控制元件,資料系結,XAML性能,自適應設計,自適應UI,自適應代碼,檔案管理,SQLite資料庫,應用程式到應用程式通信,應用程式本地化,應用程式 ......

    uj5u.com 2023-04-20 07:22:35 more