求下面函数的返回值(微软)

int func(x) 

    int countx = 0; 
    while(x) 
    { 
          countx ++; 
          x = x&(x-1); 
     } 
    return countx; 
}

假定x = 9999。 答案:8

思路:将x转化为2进制,看含有的1的个数。

求下面函数的返回值(微软) -- 统计1的个数
-------------------------------------
int func(int x)
{
    int countx = 0;
    while(x)
    {
        countx++;
        x = x&(x-1);
    }
    return countx;
}

假定x = 9999
10011100001111
答案: 8

思路: 将x转化为2进制,看含有的1的个数。
注: 每执行一次x = x&(x-1),会将x用二进制表示时最右边的一个1变为0,因为x-1将会将该位(x用二进制表示时最右边的一个1)变为0。

判断一个数(x)是否是2的n次方?
-------------------------------------
#include <stdio.h>

int func(int x)
{
    if( (x&(x-1)) == 0 )//只有二进制含有1个1的时候,x是2 的n次方!!!
        return 1;
    else
        return 0;
}

int main()
{
    int x = 8;
    printf("%d\n", func(x));
}

注: 
(1) 如果一个数是2的n次方,那么这个数用二进制表示时其最高位为1,其余位为0。
(2) == 优先级高于 &

网友1:

x=x&(x-1) 
============== 
以前没有见过这样的表达式,分析一下发现发明这个表达式的人是个高手。 
表达式的意思就是把x的二进制表示从最低位直到遇到第一个1的比特置0。 
例如: 
e1: 
x          =   01001000 
x-1       =   01000111 
x&(x-1)=01000000 
e2: 
x           =   01001001 
x-1       =   01001000 
x&(x-1)=01001000
 
我的总结:
就用e1作为例子
第一次位于运算的结果是01000000
那么继续x-1,则x-1=00111111
那么继续位于运算:
01000000&00111111=00000000
此时循环的变量x为0,循环终止。
用来计数的countx也就是说明了二进制数种有几个1.
但是我们传进去的参数是整数,例如传9999呢?
看到&运算,肯定是变成二进制算,所以传什么就是多虑了。
 
 
 
网友2:
位运算。 
&是位与。
我觉得楼主应该知道&这个是位运算中的位与运算, 
楼主只是想知道x=x&(x-1)到底有什么功能~~
网友3:
<Hacker 's   Delight> 这本书第1章就有介绍,这种算法是把一个二进制数最右边的一个1变成0。

一个简单的程序,先给出运算结果:

1011001 
1011000 
1010000 
1000000

这是代码:

#include   <stdio.h>

void   OutBin(int   n) 

int   a[32],   l   =   0;

if(n   ==   0)   { 
printf( "0 "); 
return; 
}

while(n   >   0)   { 
a[l++]   =   n   %   2; 
n   > > =   1; 
}

while(l--)   
printf( "%d ",   a[l]); 
}

int   main() 

int   x   =   89;

OutBin(x); 
printf( "\n ");

x   &=   x-1;

OutBin(x); 
printf( "\n ");

x   &=   x-1;

OutBin(x); 
printf( "\n ");

x   &=   x-1;

OutBin(x); 
printf( "\n ");

return   0; 
}

 
 
网友4:
那可以通过调试找出一点规律~~~ 
当   x   =   0,结果为0 
<0,0> , <1,0> , <2,0> , <3,2> , <4,0> , <5,4> , <6,4> , <7,6> , <8,0> , <9,8> , <10,8> , <11,10> , <12,8> , <13,12> , <14,12> , <15,14> , <16,0> , <17,16> , <18,16> , <19,18> , <20,16> , <21,16> , <22,20> , <23,22> , <24,16> , <25,24> , <26,24> , <27,26> , <28,24> , <29,28> , <30,28> , <31,30> , <32,0> ... 
所以得出结论为: 
当x为奇数的时候,x=x&(x-1)它的值相当于x   =   x-1;一样的效果~~~ 
当x为2的N次幂时,结果为0; 
其他希望下面人能找出一点规律出来~~~
 
 
网友5:
位运算里有学问呀, 
例如众所周知的交换算法: 
void   swap(int   i1,   int   i2) 

        i1   ^=   i2; 
        i2   ^=   i1; 
        i1   ^=   i2; 

还有,我今天看了Minix操作系统作者写的《操作系统   设计与实现》(写的比William   Stalling的《操作系统   内核与设计原理》有条理而且清晰紧凑得多,后者内容芜杂)中的页面替换算法之一矩阵法,就是用位运算实现的: 
假设内存分为n页,那么高速缓存一个n   x   n的比特矩阵,开始时全置0,如下(假设n=4): 
    0   1   2   3 
0   0   0   0   0 
1   0   0   0   0 
2   0   0   0   0 
3   0   0   0   0 
每次内存访问时,如果访问的是i页,那么先把矩阵的第i行置1,然后把矩阵的第i列置0,这样i行的二进制的值越小就表示i页最长时间最近没有被访问。例如假设访问的次序为0-2-3-1,那么该矩阵的变化过程为: 
    0   1   2   3 
0   0   1   1   1         0   1   0   1         0   1   1   0         0   0   1   0 
1   0   0   0   0         0   0   0   0         0   0   0   0         1   0   1   1 
2   0   0   0   0         1   1   0   1         1   1   0   0         1   0   0   0 
3   0   0   0   0         0   0   0   0         1   1   1   0         1   0   1   0 
第三个例子是Windows   GDI的二元和三元光栅操作的编码。比较复杂,就不讲了。
更正: 
void   swap(int   i1,   int   i2) 
改成: 
void   swap(int&   i1,   int&   i2)

二进制x&(x-1);的更多相关文章

  1. 使用struct处理二进制

    有的时候需要用python处理二进制数据,比如,存取文件.socket操作时.这时候,可以使用python的struct模块来完成. struct模块中最重要的三个函数是pack(), unpack( ...

  2. 如何开启MySQL 5.7.12 的二进制日志

    1. 打开/etc下的my.cnf文件 2. 编辑它,添加内容: log_bin=binary-log   #二进制日志的文件名 server_id=1  #必须指定server_id,这是MySQL ...

  3. 【.net 深呼吸】使用二进制格式来压缩XML文档

    在相当多的情况下,咱们写入XML文件默认是使用文本格式来写入的,如果XML内容是通过网络传输,或者希望节省空间,特别是对于XML文档较大的情况,是得考虑尽可能地压缩XML文件的大小. XmlDicti ...

  4. Javascript的二进制数据处理学习 ——nodejs环境和浏览器环境分别分析

    以前用JavaScript主要是处理常规的数字.字符串.数组对象等数据,基本没有试过用JavaScript处理二进制数据块,最近的项目中涉及到这方面的东西,就花一段时间学了下这方面的API,在此总结一 ...

  5. 浅析MySQL基于ROW格式的二进制日志

    上文分析的二进制日志实际上是基于STATEMENT格式的,下面我们来看看基于ROW格式的二进制日志,毕竟,两者对应的binlog事件类型也不一样,同时,很多童鞋反映基于ROW格式的二进制日志无法查到原 ...

  6. 浅析MySQL二进制日志

    查看MySQL二进制文件中的内容有两种方式 1.  mysqlbinlog 2.  SHOW BINLOG EVENTS [IN 'log_name'] [FROM pos] [LIMIT [offs ...

  7. asp.net将图片转成二进制存入数据库

    一.代码如下 int code = int.Parse(this.TextBox1.Text);//图片编码 string value = this.FileUpload1.PostedFile.Fi ...

  8. 二进制包安装MySQL数据库

    1.1二进制包安装MySQL数据库 1.1.1 安装前准备(规范) [root@Mysql_server ~]# mkdir -p /home/zhurui/tools ##创建指定工具包存放路径 [ ...

  9. sqlite3的图片的(二进制数据)存取操作

    sqlite3的图片的(二进制数据)存取操作   前言 上篇介绍了sqlite3的一些常用插入操作方法和注意事项,在实际项目中遇到了图片缓存的问题,由于服务器不是很稳定,且受到外界环境的干扰(例如断电 ...

  10. MySQL二进制日志

    一.二进制日志(The Binary Log) 1.简介 包含所有更新了的数据或者已经潜在更新了的数据(比如一条没有匹配任何行的delete语句) 包含所有更新语句执行时间的信息 不记录没有修改数据的 ...

随机推荐

  1. mysql -- 预处理语句

    所谓预处理,即在真正执行某条SQL语句之前,先将SQL语句准备好,在执行过程中再绑定数据 语法: 准备预处理 prepare 预处理名字 from ‘要执行的SQL语句’ 执行预处理 execute ...

  2. 【wikioi】3160 最长公共子串(后缀自动机)

    http://codevs.cn/problem/3160/ sam的裸题...(之前写了spoj上另一题sam的题目,但是spoj被卡评测现在还没评测完QAQ打算写那题题解时再来详细介绍sam的.. ...

  3. MD5的各种实现——也是醉了

    MD5即Message-Digest Algorithm 5(信息-摘要算法5).用于确保信息传输完整一致. 是计算机广泛使用的杂凑算法之中的一个(又译摘要算法.哈希算法),主流编程语言普遍已有MD5 ...

  4. db2 将逗号分隔数据转换为多值IN列表

    将逗号分隔数据转换为多值IN列表 2010-03-15 11:16:59|  分类: 数据库技术|举报|字号 订阅     下载LOFTER我的照片书  |     原文:http://book.cs ...

  5. 打打基础,回头看看avr单片机的定时器、中断和PWM(转)

    以前小看了定时器,发现这东西还真的很讲究,那先复习复习吧. 先提提中断:我的理解就是cpu执行时,遇到中断——根据对应的中断源(硬件或软件)——pc定位中断入口地址,然后根据这里的函数指针——跳转到相 ...

  6. Zabbix-3.0.3使用自带模板监控MySQL

    导读 Zabbix是一款优秀的,开源的,企业级监控软件,可以通过二次开发来监控你想要监控的很多服务,本文介绍使用Zabbix自带的模板监控MySQL服务. 配置userparameter_mysql. ...

  7. 【BZOJ3781、2038】莫队算法2水题

    [BZOJ3781]小B的询问 题意:有一个序列,包含N个1~K之间的整数.他一共有M个询问,每个询问给定一个区间[L..R],求Sigma(c(i)^2)的值,其中i的值从1到K,其中c(i)表示数 ...

  8. LeetCode 笔记系列 14 N-Queen II [思考的深度问题]

    题目: Follow up for N-Queens problem. Now, instead outputting board configurations, return the total n ...

  9. Vue基础-渲染函数-父子组件-传递数据

    Vue 测试版本:Vue.js v2.5.13 做了个 demo,把父子组件的数据都绑定到 Vue 实例 app 上,注释中的 template 相对好理解些 <div id="app ...

  10. 170502、linux下配置jdk8

    1.下载源码包jdk-8u45-linux-x64.tar.gz(省略) 2.解压 tar -zxvf jdk-8u45-linux-x64.tar.gz 3.配置jdk vi /etc/profil ...