009-NumberOf1InBinary-【c语言】

18 篇文章 0 订阅


1、题目

输入一个整数,输出该数二进制表示中1的个数。负数计算补码中1的个数(假设该机器是32bit的)

2、分析

【思路一】

众所周知:1 & n(n表示任意整数)会得到 n 的最后一位二进制位(0或1)
所以我们只需要控制 1 和 n 中的所有二进制位进行 & 运算就能知道对应位置上是 0 还是 1

示例:1
00000000 00000000 00000000 00001101 是 十进制数 13 的原码、反码和补码
令 n = 00000000 00000000 00000000 00001101
(n >> 0) & 1 = 1 - 得到了n中第一位二进制位,即1(从右往左数)
(n >> 1) & 1 = 0 - 得到了n中第二位二进制位,即0(从右往左数)
(n >> 1) & 1 = 1 - 得到了n中第三位二进制位,即1(从右往左数)
. . . . . .
(n >> 30) & 1 = 0 - 得到了n中第三十一位二进制位,即0(从右往左数)
(n >> 31) & 1 = 0 - 得到了n中第三十二位二进制位,即0(从右往左数)

示例2:
令 n = -13
n的原码为:10000000 00000000 00000000 00001101
n的反码为:11111111 11111111 11111111 11110010
n的补码为:11111111 11111111 11111111 11110011
再令 n = 11111111 11111111 11111111 11110011
(n >> 0) & 1 = 1 - 得到了n中第一位二进制位,即1(从右往左数)
(n >> 1) & 1 = 1 - 得到了n中第二位二进制位,即1(从右往左数)
(n >> 2) & 1 = 1 - 得到了n中第三位二进制位,即0(从右往左数)
. . . . . .
(n >> 30) & 1 = 1 - 得到了n中第三十一位二进制位,即1(从右往左数)
(n >> 31) & 1 = 1 - 得到了n中第三十二位二进制位,即1(从右往左数)

我们不难看出,在写代码时需要使用一个变量来控制移动的偏移量,假设这个变量是 i ,那么 i 从0开始,到31结束。

【思路二】

这个方法如果你没有做过类似的题那么你将会很南想到这个方法!!!
方法:n & (n - 1) 该公式每次都会消除n中二进制为的一个1,当然是先消除低地址的。

示例:
令 n = 13
n的原码、反码和补码均为:00000000 00000000 00000000 00001101
第一次:
n = n & ( n - 1 ) = 13 & 12
即 n = 1101 & 1100 = 1100 - 此时的 n 被更新为 1100 了,可以很清楚的看到,它和上一次的n二进制数相比,的确是在低地址处少了一个 1
二进制 1100 对应十进制的 12
第二次:
n = n & ( n - 1) = 12 & 11
即 n = 1100 & 1011 = 1000 - 此时的 n 被更新为 1100 了,可以很清楚的看到,它和上一次的n二进制数相比,在低地址处也少了一个 1
二进制 1000 对应十进制的 8
第三次:
n = n & ( n - 1) = 8 & 7
即 n = 1000 & 0111 = 0
此时已经运算已经结束了,因为 n = 0 了,再运算下去已经没有意义了

可以清楚的看到,13的二进制位中有三个1,而我们也进行了三次运算,这并不是巧合,大家自己下去可以多用几个数字尝试一下。

3、代码实现

【思路一代码】

int numberOf1(int n)
{
	// 记录1的数量
	int count = 0;
	// 记录每位二进制位
	int temp = 0;
	int i = 0;

	for ( i = 0; i < 32; i ++)
	{
		temp = n >> i;
		if ( (temp & 1) ) count ++;
	}

	return count;
}

【思路二代码】

int numberOf1(int n)
{
	// 记录1的数量
	int count = 0;
	while ( n )
	{
		n = n & (n-1);
		count ++;
	}
	return count;
}

文章索引

调整奇偶数位置
模拟实现字符串长度函数 strlen
手写顺序栈
2的n次方


后记

我水平有限,错误难免,还望各位加以指正。


本文内容到此结束,感谢您的阅读!!!如果内容对你有帮助的话,记得给我三连丫(点赞、收藏、关注)


本人博客所有文章,均为原创。部分文章中或引用相关资料,但均已著明来源出处。可随意转载、分享,但需加本文链接,以及版权说明。

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值