C语言判断一个数是否是2的幂次方或4的幂次方

快速判断一个数是否是2的幂次方,若是,并判断出来是多少次方!
将2的幂次方写成二进制形式后,很容易就会发现有一个特点:二进制中只有一个1,并且1后面跟了n个0; 因此问题可以转化为判断1后面是否跟了n个0就可以了。

如果将这个数减去1后会发现,仅有的那个1会变为0,而原来的那n个0会变为1;因此将原来的数与去减去1后的数字进行与运算后会发现为零。

最快速的方法:

   (number & number - 1) == 0

原因:因为2的N次方换算是二进制为10……0这样的形式(0除外)。与上自己-1的位数,这们得到结果为0。例如。8的二进制为1000;8-1=7,7的二进制为111。两者相与的结果为0。计算如下:

     1000
   & 0111
    -------
    0000

使用递归来实现的代码如下:

#include "stdio.h"
#include "stdlib.h" 

int log2(int value)  //递归判断一个数是2的多少次方
{
  if (value == 1)
    return 0;
  else
    return 1+log2(value>>1);
} 

int main(void)
{
  int num;
  printf("请输入一个整数:");
  scanf("%d",&num);
  if(num&(num-1)) //使用与运算判断一个数是否是2的幂次方
    printf("%d不是2的幂次方!\n",num);
  else
    printf("%d是2的%d次方!\n",num,log2(num));
  system("pause");
  return 0;
}

使用非递归来实现的代码如下:

#include "stdio.h"
#include "stdlib.h" 

int log2(int value)  //非递归判断一个数是2的多少次方
{
  int x=0;
  while(value>1)
  {
    value>>=1;
    x++;
  }
  return x;
} 

int main(void)
{
  int num;
  printf("请输入一个整数:");
  scanf("%d",&num);
  if(num&(num-1))   //使用与运算判断一个数是否是2的幂次方
    printf("%d不是2的幂次方!\n",num);
  else
    printf("%d是2的%d次方!\n",num,log2(num));
  system("pause");
  return 0;
}

扩展:求一个数n的二进制中1的个数。
非常巧妙地利用了一个性质,n=n&(n-1) 能移除掉n的二进制中最右边的1的性质,循环移除,直到将1全部移除,这种方法将问题的复杂度降低到只和1的个数有关系。代码如下:

int Func3(int data)
{  //利用了data&(data-1)每次都能移除最右边的1,移除了多少个1,就是包含了几个1
  int count = 0;
  while (data)
  {
    data = data & (data-1);
    count++;
  }
  return count;
}

扩展问题二:

A和B的二进制中有多少位不相同。这个问题可以分为两步,(1)将A和B异或得到C,即C=A^B,(2)计算C的二进制中有多少个1。


快速判断一个数是否是4的幂次方,若是,并判断出来是多少次方!
将4的幂次方写成二进制形式后,很容易就会发现有一个特点:二进制中只有一个1(1在奇数位置),并且1后面跟了偶数个0; 因此问题可以转化为判断1后面是否跟了偶数个0就可以了。

4的整数次幂的二进制数都为 (4)100、(16)10000、(64)1000000......

另外,4的幂次方4^n也可以写为2^(2*n),即也可以写为2的幂次方,当然就满足2的幂次方的条件了,即num & num-1==0。

思路:首先用条件num & num-1==0来判断是否为2的幂次方,若不满足,则不是。若满足,在用条件num & 0x55555555来判断,若为真,则这个整数是4的幂次方,否则不是。

使用递归来实现的代码如下:

#include "stdio.h"
#include "stdlib.h" 

bool fn(unsigned int x)   //判断x是否是4的幂次方
{
 if ( x & (x - 1) )     //判断x是否为2的幂次方
   return false;
 return x & 0x55555555;   //判断1是否在奇数位置上
} 

int log4(int value)   //递归判断一个数是4的多少次方
{
  if (value == 1)
    return 0;
  else
  {
    value>>=1;    //往右移位
    return 1+log4(value>>1);    //往右移位
  }
} 

int main(void)
{
  int num;
  printf("请输入一个整数:");
  scanf("%d",&num);
  if(fn(num))   //使用与运算判断一个数是否是2的幂次方
    printf("%d是4的%d次方!\n",num,log4(num));
  else
    printf("%d不是4的幂次方!\n",num);
  system("pause");
  return 0;
}

使用非递归来实现的代码如下:

#include "stdio.h"
#include "stdlib.h" 

bool fn(unsigned int x)   //判断x是否是4的幂次方
{
 if ( x & (x - 1) )     //判断x是否为2的幂次方
   return false;
 return x & 0x55555555;   //判断1是否在奇数位置上
} 

int log4(int value)  //非递归判断一个数是4的多少次方
{
  int x=0;
  while(value>1)
  {
    value>>=1;   //往右移位
    value>>=1;
    x++;
  }
  return x;
}  

int main(void)
{
  int num;
  printf("请输入一个整数:");
  scanf("%d",&num);
  if(fn(num))   //使用与运算判断一个数是否是2的幂次方
    printf("%d是4的%d次方!\n",num,log4(num));
  else
    printf("%d不是4的幂次方!\n",num);
  system("pause");
  return 0;
}
(0)

相关推荐

  • C语言快速幂取模算法小结

    本文实例汇总了C语言实现的快速幂取模算法,是比较常见的算法.分享给大家供大家参考之用.具体如下: 首先,所谓的快速幂,实际上是快速幂取模的缩写,简单的说,就是快速的求一个幂式的模(余).在程序设计过程中,经常要去求一些大数对于某个数的余数,为了得到更快.计算范围更大的算法,产生了快速幂取模算法.我们先从简单的例子入手:求abmodc 算法1.直接设计这个算法: int ans = 1; for(int i = 1;i<=b;i++) { ans = ans * a; } ans = ans %

  • C语言求幂计算的高效解法

    本文实例演示了C语言求幂计算的高效解法.很有实用价值.分享给大家供大家参考.具体方法如下: 题目如下: 给定base,求base的幂exp 只考虑基本功能,不做任何边界条件的判定,可以得到如下代码: #include <iostream> using namespace std; int cacExp(int base, int exp) { int result = 1; int theBase = 1; while (exp) { if (exp & 0x01) result =

  • 用C语言求幂函数和指数函数的方法

    C语言pow()函数:求x的y次方(次幂) 头文件: #include <math.h> pow() 函数用来求 x 的 y 次幂(次方),其原型为: double pow(double x, double y); pow()用来计算以x 为底的 y 次方值,然后将结果返回.设返回值为 ret,则 ret = xy. 可能导致错误的情况: 如果底数 x 为负数并且指数 y 不是整数,将会导致 domain error 错误. 如果底数 x 和指数 y 都是 0,可能会导致 domain err

  • C语言判断一个数是否为素数方法解析

    一.概念介绍 素数又称为质数.一个大于1的自然数(从2开始),除了1和它本身外,不能被其他自然数整除的叫做素数,否则称为合数. 0和1既不是素数也不是合数,最小的素数是2. 二.代码 方法一: bool is_Prime(int num){ int i; for(i = 2;i <= sqrt(num);i++){ if(num % i == 0)//一旦发现有因子,则返回false return false; } return true; } 注意:在for循环判断时不能忘记 i <= sq

  • C语言判断一个数是否是2的幂次方或4的幂次方

    快速判断一个数是否是2的幂次方,若是,并判断出来是多少次方! 将2的幂次方写成二进制形式后,很容易就会发现有一个特点:二进制中只有一个1,并且1后面跟了n个0: 因此问题可以转化为判断1后面是否跟了n个0就可以了. 如果将这个数减去1后会发现,仅有的那个1会变为0,而原来的那n个0会变为1:因此将原来的数与去减去1后的数字进行与运算后会发现为零. 最快速的方法: (number & number - 1) == 0 原因:因为2的N次方换算是二进制为10--0这样的形式(0除外).与上自己-1的

  • Python编程学习之如何判断3个数的大小

    前言 大部分初学编程的人来说刚开始都会练习判断两个数或者三个数的大小,来熟悉某种语言的特性和最基本的if,else循环,当我们学习了更高级的语法知识后,又会有不同的实现方式,比如这道练习题依次接收用户输入的3个数,排序后打印现在我们来看一下在Python中都有哪些方法来实现: 1, 采用分支结构,用最基本的if和else来实现: a = int(input('a>>>')) b = int(input('b>>>')) c = int(input('c>>&

  • C语言判断数是否为素数与素数输出

    目录 1.判断单个数是否为素数(多组输入) 2.输入范围输出范围内的素数 3.总结 素数的概念:素数也叫质数,是一种只能被自己本身和1整除的数并且大于1,当然0与1不是素数. 1.判断单个数是否为素数(多组输入) 我的思路是,首先输入一个数,利用素数的概念来判断是非为素数,是的话输出素数:否则不输出. 关于素数的判断首先我们吧输入的数当初被除数,我选择用一个for循环来实现,从2开始当除数,每轮加1,一直循环去除被除数,一直除到被除数减一那个数,要是期间能被一个数整除则跳出循环不为素数,要是一直

  • Go语言判断指定文件是否存在的方法

    本文实例讲述了Go语言判断指定文件是否存在的方法.分享给大家供大家参考.具体实现方法如下: 复制代码 代码如下: package main    import (     "fmt"     "os" )    func main() {     f, err := os.Open("dotcoo.com.txt")     if err != nil && os.IsNotExist(err) {         fmt.Pri

  • 使用C语言判断栈的方向实例

    这一问题主要是如何判读出先后入栈的变量的地址大小,比如有a, b两个变量一先一后被定义,如果a的地址大于b的地址,则说明是以低地址方向增长的,反之,往高地址方向增长.在写C程序的时候不能简单直接的定义两个变量来比较它们的地址大小,因为这样很有可能编译器会做优化,最终导致结果不真实.为避免这种编译器优化的情况,可以采用将变量定义到函数中,然后递归调用该函数. 例如下面的代码: #include <stdio.h> static int stack_direction = 0; static vo

  • JS判断一个数是否是水仙花数

    水仙花数是指一个 n 位数 ( n≥3 ),它的每个位上的数字的 n 次幂之和等于它本身. 例如:1^3 + 5^3+ 3^3 = 153 //判断一个数是否数水仙花数 var num=prompt('请输入一个数字'); //得到位数可以计算幂数 var length=num.length; //使用字符串的方法获取每一位数 var content=num.split(""); //判断开始输入的数字和计算出来的结果是否相等 var result=0; for(var i=0;i&l

  • Go语言判断文件或文件夹是否存在的方法

    本文实例讲述了Go语言判断文件或文件夹是否存在的方法.分享给大家供大家参考,具体如下: Golang 判断文件是否存在有点怪异,是根据在操作文件时返回的错误信息来判断的,而不能直接根据路径判断 版本1: 复制代码 代码如下: func IsExists(path string) (bool, error) {     _, err := os.Stat(path)     if err == nil {         return true, nil     }     if os.IsNot

  • 判断一个数是不是素数的方法

    给出一个数,判断这个数是不是素数: 复制代码 代码如下: #include <cmath> bool isPrime(int n) {  int i;  for (i = 2; i <= sqrt(n); i++) {    if (n % i == 0)      return false;  }  return true;}

  • 使用c语言判断100以内素数的示例(c语言求素数)

    从console输入一个数,判断这个数是否为素数(质数). 复制代码 代码如下: #include <stdio.h> /**判断100以内的素数*/ //定义函数判断是否是素数int isPrime(int num ){    int i;    //从2开始循环,一直到i的平方小于等于给定的数.    for (i = 2; i*i <= num; i++) {        if ( ( num % i ) == 0 ) {            return 0;       

随机推荐