判断是否是2的N次方——证明x & (x - 1)==0的正确性

本文介绍了一种判断整数是否为2的N次方的有效方法:通过按位与运算X&(X-1)来判断。文章详细证明了该方法的正确性,并通过数学逻辑解释了其背后的原理。

  判断一个整数x是否是2的N次方。

  方法之一是判断x & (x - 1)==0。若为True,则x是2的N次方;若为False,则x不是2的N次方。

  有人质疑,他证明了“2的n次方一定符合这个条件”, 却并没有证明“符合这个条件的一定是2的n次方”呀!更没有证明“不符合条件的一定不是2的n次方”呀。

  

  现在,从两个方面来证明这个方法的正确性

 证明之前,先给出一些定义

  &运算的定义:A & B 表示将A和B转化为二进制,然后按照对位&运算。

  例如:17 & 9

    100012  =1710

  &     1012   =910

  ------------------------

    000012   =110

  而对位&运算的定义如下:

  1 & 1=1  ;  1 & 0=0  ;  0 & 1=0  ;  0 & 0=0

  对位&运算还有如下性质:

  A & 1=A  ;  A & 0=0  ;  A & A=A  ;  A & B=B & A  此时:A,B=0或1

  

  定义:

  X=x1x2……xn-1xn,其中xi=1或0,1≤i≤n,n>0。显然X>0(当X≤0,没有讨论的意义)

  给定正整数X,X是2的N次方的充要条件是X转化成二进制后,有且只能有一个1,其余的都是0

  也就是说,若X是2的N次方,则x1=1,x2=……=xn-1=xn=0

       若X不是2的N次方,则至少存在一个j,xj=1,1<j≤n

  

 先证明“2的N次方符合X & (X - 1)==0条件”

  当X=1时,1 & 0 =0,满足条件

  当X>1时,且X是2的N次方

  如定义:X=100……0  (n-1个0,n>1)

      X-1=11……1  (n-1个1,n>1)

  则X & X-1是

     100……02  =X10      

  &     11……12  =X-110

  ------------------------

      00……02  =010

 

  满足条件 

 

 再证明“不是2的N次方不符合X & (X - 1)==0条件”

  分两种情况,

  1、X是奇数,则X=x1x2……xn-1xn,x1=xn=1,故X=1x2……xn-11

    则X-1=1x2……xn-10

    则X & X-1是

     1x2x3……xn-112  =X10      

  &     1x2x3……xn-102  =X-110

  ------------------------------------

     1x2x3……xn-102   ≠010

     不满足X & (X - 1)==0的条件

  2、X是偶数,则X=x1x2……xn-1xn,x1=1,xn=0

    由于X不是2的N次方,因此x1,x2……xn-1中至少有两个为1。设xj是最右边的1

    则X=1x2……xj-1xj0……0=1x2……xj-110……0   1<j<n,最右边有n-j个0

    则X-1=1x2……xj-101……1           1<j<n,最右边有n-j个1

    则X & X-1

     1x2……xj-110……02  =X10      

  &     1x2……xj-101……12  =X-110

  --------------------------------------

     1x2……xj-100……02   ≠010

    不满足X & (X - 1)==0的条件  

  综上所述,当X不是2的N次方的时候,是不满足X & (X - 1)==0的条件的

 

  因此,当X是2的N次方的时候X & (X - 1)==0成立,当X不是2的N次方的时候X & (X - 1)==0不成立。

 

  故判断X(X>0)是否是2的N次方的方法,判断X & (X - 1)==0是否成立,是可行的。


作者: 万仓一黍

### 表达式 `x&amp;(x-1)!=0` 的逻辑意义与计算结果 #### 逻辑意义 表达式 `x&amp;(x-1)` 的作用是清除整数 `x` 的二进制表示中最右侧的 `1`。如果 `x` 的二进制表示中存在至少一个 `1`,那么 `x&amp;(x-1)` 的结果将不等于 `x`[^4]。因此,`x&amp;(x-1)!=0` 的逻辑意义是判断整数 `x` 是否为非零值且其二进制表示中存在多个 `1`。 具体来说: - 如果 `x` 是 **2 的幂方**(即二进制表示中只有一个 `1`),那么 `x&amp;(x-1)` 的结果为 `0`,此时 `x&amp;(x-1)!=0` 的结果为假。 - 如果 `x` 不是 **2 的幂方**(即二进制表示中有多个 `1`),那么 `x&amp;(x-1)` 的结果不为 `0`,此时 `x&amp;(x-1)!=0` 的结果为真。 #### 计算结果 以下通过几个示例说明 `x&amp;(x-1)!=0` 的计算结果: 1. 当 `x = 8`(二进制为 `1000`)时: ```c x = 8; // 二进制为 1000 x - 1 = 7; // 二进制为 0111 x &amp; (x - 1) = 1000 &amp; 0111 = 0000; ``` 因此,`x&amp;(x-1)!=0` 的结果为假。 2. 当 `x = 6`(二进制为 `0110`)时: ```c x = 6; // 二进制为 0110 x - 1 = 5; // 二进制为 0101 x &amp; (x - 1) = 0110 &amp; 0101 = 0100; ``` 因此,`x&amp;(x-1)!=0` 的结果为真。 3. 当 `x = 0` 时: ```c x = 0; x - 1 = -1; // 假设使用补码表示,则为 ...11111111 x &amp; (x - 1) = 0 &amp; ...11111111 = 0; ``` 因此,`x&amp;(x-1)!=0` 的结果为假。 #### 示例代码 以下是一个完整的代码示例,展示如何使用 `x&amp;(x-1)!=0` 判断整数 `x` 是否满足条件: ```c #include &lt;stdio.h&gt; int main() { int x; while (scanf(&quot;%d&quot;, &amp;x) != EOF) { if (x == 0) { printf(&quot;x为0\n&quot;); } else if ((x &amp; (x - 1)) == 0) { printf(&quot;x为2的幂方\n&quot;); } else { printf(&quot;x不是2的幂方\n&quot;); } } return 0; } ``` #### 应用场景 表达式 `x&amp;(x-1)` 的典型应用场景包括: 1. 判断一个整数是否为 **2 的幂方**。 2. 统计整数的二进制表示中 `1` 的个数(通过不断执行 `x&amp;(x-1)` 操作,直到 `x` 变为 `0`)。 3. 在位运算优化中,快速清除最低位的 `1`。 #### 注意事项 在实际应用中需要注意以下几点: - 对于负数,`x&amp;(x-1)` 的行为依赖于编译器实现,可能不符合预期。 - 在多线程环境中,直接对共享变量执行 `x&amp;(x-1)` 操作可能导致竞态条件。 ---
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值