行文脉络

  1. 解法一——除法
  2. 解法二——移位
  3. 解法三——高效移位
  4. 解法四——查表
  5. 扩展问题——异或后转化为该问题

对于一个字节(8bit)的变量,求其二进制“1”的个数。例如6(二进制0000 0110)“1”的个数为2,要求算法效率尽量高。

解法一

对于二进制数来说,除一个2,就少一位,可以判断这个少的位来确定“1”的个数。

例如:6(0000 0110)

0000 0110 / 2 = 0000 0011     ----少的一位为0

0000 0011 / 2 = 0000 0001     ----少的一位为1

0000 0001 / 2 = 0000 0000     ----少的一位为1

操作数数已经为0,到此结束

参考代码

int Count_1(int val)
{
int num = ;
while(val)
{
if(val % != ) //用取模获得去除的一位
++num;
val /= ;
}
return num;
}

性能:时间复杂度O(log2v),即二进制数的位数;空间复杂度O(1)

解法二

对于二进制数来说,除法是用移位完成。

例如:6(0000 0110)

0000 0110 >> 1 = 0000 0011     ----少的一位为0

0000 0011 >> 1 = 0000 0001     ----少的一位为1

0000 0001 >> 1 = 0000 0000     ----少的一位为1

操作数数已经为0,到此结束

参考代码

int Count_2(int val)
{
int num = ;
while(val)
{
if(val & != ) //用与1与获得移除的一位
++num;
val >>= ;
}
return num;
}

性能:时间复杂度O(log2v),即二进制数的位数;空间复杂度O(1)

解法三

对于上述算法,有个问题,比如1000 0000,大把的时间用在没用的0上,最好寻求一种直接判断“1的个数。

通过观察可以找到规律:对于数a, a = a & (a-1)就可以去除a的最后一个1

例如:6(0000 0110)

0000 0110 & 0000 0101 = 0000 0100

0000 0100 & 0000 0011 = 0000 0000

操作数数已经为0,到此结束

参考代码

int Count_3(int val)
{
int num = ;
while(val)
{
val &= (val -);
++num;
}
return num;
}

性能:时间复杂度O(M),即二进制中“1”的个数,空间复杂度O(1)

解法四

查表法,把0~255这256个数的结果全部存储在数组中,val直接作为下标,countTable[val]即为结果。典型的用空间换时间。

参考代码

int Count_5(int val)
{
int countTable[] = {, , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , };
return countTable[val];
}

性能:时间复杂度:O(1), 空间复杂度O(N)

整个程序执行参考

#include<iostream>
using namespace std; int Count_1(int val)
{
int num = ;
while(val)
{
if(val % != )
++num;
val /= ;
}
return num;
} int Count_2(int val)
{
int num = ;
while(val)
{
if(val & != )
++num;
val >>= ;
}
return num;
} int Count_3(int val)
{
int num = ;
while(val)
{
val &= (val -);
++num;
}
return num;
}
int Count_5(int val)
{
int countTable[] = {, , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , , };
return countTable[val];
} int main()
{
int a = , b = ;
cout << "Num of 1:" << Count_1(a) << endl;
cout << "Num of 1:" << Count_2(a) << endl;
cout << "Num of 1:" << Count_3(a) << endl;
cout << "Num of 1:" << Count_5(b) << endl;
cout << "Num of 1:" << Count_1(b) << endl;
cout << "Num of 1:" << Count_2(b) << endl;
cout << "Num of 1:" << Count_3(b) << endl;
cout << "Num of 1:" << Count_5(b) << endl;
}

结果

扩展问题

1. 给定两个正整数(二进制表示)A、B,如何快速找出A和B二进制表示中不同位数的个数。

  思路

  首先A和B进行异或操作,然后求得到的结果中1的个数(此问题)。

2. 判断一个数是否是2的幂

bool powerof2(int n)
{
return ((n & (n-)) == );
}

Algorithm --> 二进制中1的个数的更多相关文章

  1. 剑指Offer面试题:9.二进制中1的个数

    一.题目:二进制中1的个数 题目:请实现一个函数,输入一个整数,输出该数二进制表示中1的个数.例如把9表示成二进制是1001,有2位是1.因此如果输入9,该函数输出2. 二.可能引起死循环的解法 一个 ...

  2. 剑指Offer:二进制中1的个数

    题目:输入一个整数,输出该数二进制表示中1的个数. // 二进制中1的个数 #include <stdio.h> int wrong_count_1_bits(int n) // 错误解法 ...

  3. 1513:二进制中1的个数 @jobdu

    题目1513:二进制中1的个数 时间限制:1 秒 内存限制:128 兆 特殊判题:否 提交:1341 解决:455 题目描述: 输入一个整数,输出该数二进制表示中1的个数.其中负数用补码表示. 输入: ...

  4. 基于visual Studio2013解决面试题之0410计算二进制中1的个数

     题目

  5. [PHP]算法-二进制中1的个数的PHP实现

    二进制中1的个数: 输入一个整数,输出该数二进制表示中1的个数.其中负数用补码表示. 思路: 1.右移位运算>> 和 与运算& 2.先移位个然后再与1 &运算为1的就是1 ...

  6. 《剑指offer》 二进制中1的个数

    本题来自<剑指offer> 二进制中1的个数 题目: 输入一个整数,输出该数二进制表示中1的个数.其中负数用补码表示. 思路: 两种思路: 第一种:对n进行左移,检测最后一位是否为1,但考 ...

  7. 剑指offer编程题Java实现——面试题10二进制中1的个数

    题目: 请实现一个函数,输入一个整数,输出该整数二进制表示中1的个数.例如,把9表示成二进制是1001,有2位是1,该函数输出2解法:把整数减一和原来的数做与运算,会把该整数二进制表示中的最低位的1变 ...

  8. leetcode 338. Counting Bits,剑指offer二进制中1的个数

    leetcode是求当前所有数的二进制中1的个数,剑指offer上是求某一个数二进制中1的个数 https://www.cnblogs.com/grandyang/p/5294255.html 第三种 ...

  9. 【Java】 剑指offer(14) 二进制中1的个数

    本文参考自<剑指offer>一书,代码采用Java语言. 更多:<剑指Offer>Java实现合集   题目 请实现一个函数,输入一个整数,输出该数二进制表示中1的个数.例如把 ...

随机推荐

  1. org.apache.subversion.javahl.ClientException: Attempted to lock an already-locked dir

    1.错误描述 org.apache.subversion.javahl.ClientException: Attempted to lock an already-locked dir svn: Co ...

  2. Linux查看系统中的每个进程

    Linux查看系统中的每个进程 youhaidong@youhaidong-ThinkPad-Edge-E545:~$ ps -A PID TTY TIME CMD 1 ? 00:00:01 init ...

  3. Linux显示用户注册名

    Linux显示用户注册名 youhaidong@youhaidong-ThinkPad-Edge-E545:~$ finger -s Login Name Tty Idle Login Time Of ...

  4. 在Visual Studio 2012中使用GSL

    1. 下载GSL http://gnuwin32.sourceforge.net/packages/gsl.htm 下载 Complete package, except sources和Source ...

  5. WebForm 生成并显示二维码

    Generate and display QRCode in WebForm. 项目引用 QRCoder生成并显示 QRCode 项目引用 QRCoder How to use QRCoder Via ...

  6. 程序员的自我救赎---12.2.3: 虚拟币交易平台(区块链) 下 【C#与以太坊通讯】

    <前言> (一) Winner2.0 框架基础分析 (二)PLSQL报表系统 (三)SSO单点登录 (四) 短信中心与消息中心 (五)钱包系统 (六)GPU支付中心 (七)权限系统 (八) ...

  7. mfc100u.dll下载和使用方法

      当运行软件或游戏时,系统提示"丢失mfc100u.dll"."没有找到mfc100u.dll"等类似错误信息. 请下载本站提供的dll文件,使用它可以帮助用 ...

  8. 【BZOJ1212】L语言(AC自动机)

    [BZOJ1212]L语言(AC自动机) 题面 BZOJ 题解 很自然的,既然要匹配单词,那就全部都丢到\(AC\)自动机里面去 现在想想怎么匹配 先是\(AC\)自动机正常的匹配 如果此时这个位置能 ...

  9. 【noip模拟】修长城

    Time Limit: 1000ms    Memory Limit: 256MB Description 大家都知道,长城在自然条件下会被侵蚀,因此,我们需要修复.现在是21世纪,修复长城的事情当然 ...

  10. sharepoint 2013实践

    之前在一篇文章中说过了SharePoint环境的安装.那么如何使用SharePoint开发一个站点呢?这就是本篇所要阐述的问题. 在如何具体操作之前,我们先来普及下SharePoint基础知识.Far ...