zoukankan      html  css  js  c++  java
  • leetcode-201-数字范围按位与

    题目描述:

    给定范围 [m, n],其中 0 <= m <= n <= 2147483647,返回此范围内所有数字的按位与(包含 m, n 两端点)。

    示例 1: 

    输入: [5,7]
    输出: 4

    示例 2:

    输入: [0,1]
    输出: 0

     

    要完成的函数:

    int rangeBitwiseAnd(int m, int n) 

    说明:

    1、这道题给定两个int型整数,一个是开端,一个是末端,要求把开端和末端之间的数每一个都进行“与”操作,返回最后的结果。

    2、这道题肯定不是直接做一个循环,每一个都去“与”一遍,这样子太费时间。我们要从数位的角度来考虑,因为数位只有32位,更加好操作。

    如果只有两个数字,那么最后一位必然要改变,肯定一个是0,一个是1,那么与的结果肯定是0。

    如果只有三个数字,那么最后一位和倒数第二位必然要改变,因为最后一位只能存储两个数字,三个数字的话必然倒数第二位也要改变,那么这时候倒数两个数字与的结果肯定是0。

    如果有五个数字,那么最后一位、倒数第二位和倒数第三位必然要改变,因为最后两位只能存储四个数字,五个数字的话必然倒数第三位也要改变,所以最后三位与的结果肯定是0。

    所以我们可以得出规律:

    最后一位只能存储两个数,所以如果有三个数字,那么必然倒数第二位和最后一位为0。

    倒数两位只能存储四个数,所以如果有五个数字,那么必然倒数三位都为0。

    倒数三位只能存储八个数,所以如果有九个数字,那么必然倒数四位都为0。

    ……

    所以我们可以根据给出来的开端和末端,计算出中间一共有多少个数,对应地找一下应该倒数几位都为0。

    比如5是0101,7是0111,中间有三个数字5/6/7,那么必然倒数两位都是0,所以最终结果是0100。

    这道题到这里看起来似乎是解决了,但其实我们只解决了其中一种情况。

    还是上面这个例子,我们有三个数字,所以最后一位和倒数第二位都会改变,但是倒数第三位会不会改变呢?甚至倒数第四位会不会改变呢?

    我们来看一下:

    0100

    0101

    0110

    0111

    -------

    1000

    1001

    如果从0100开始,到0110结束,三个数字,倒数两位是必须改变的,倒数第二位必须0和1都有出现,然后相与为0,但是倒数第三位,由于这个时候没有发生进位,所以没有影响,我们只需要考虑倒数两位的改变。

    但是如果从0110开始,到1000结束呢,同样三个数字,同样的倒数两位必须改变,因为倒数两位的变化还是10->11->00,必然还是出现0和1,导致相与结果为0,但是这个时候前面几位却发生了进位,导致相与的结果不能是0100,而是0000。

    这时候有一种很直觉的做法,就是把开端和末端两个数字“与”一下,接着再做上面的操作——找到倒数几位必须要改为0。

    这种操作可以ac所有的测试样例,要解释的话也不难(不感兴趣原因的同学直接看代码啦),比如7和9

    00111

    01001

    这两个是“跨域”的操作,三个数照理来说只有倒数两位要改变,前面的三位是不应该变化的(如果在同一个域中),由于倒数第二位产生了进位符号,传递给了倒数第三位,导致产生了前后两种不同的前三位的表示。开端和末端的前三位的表示可以代表两种不同的状态,并且所有中间值的前三位只有这两种状态,不应该再改变了。

    这样解释可能比较难以理解,不懂的同学自己举一些需要“跨域”的例子,比如有三个数的,五个数的,甚至九个数的,多想想应该也就会比较清楚。

    代码如下:(附详解)

        int rangeBitwiseAnd(int m, int n) 
        {
            if(m==n)return m;//边界情况,开端和末尾是同一个数,直接返回
            if(m==0)return 0;//边界情况,开端是0,相与必然为0,直接返回0
            int geshu=n-m,t=log10(geshu)/log10(2)+1,t1=t;//t表示倒数的几位要置零,t1记录一下t的值
            m=m&n;//开端“与”末尾,预防“跨域”的情况
            while(t--)//把倒数t位置零,先不断右移
                m>>=1;
            while(t1--)//再不断左移
                m<<=1;
            return m;//返回m
        }

    上述代码实测24ms,beats 97.21% of cpp submissions。

  • 相关阅读:
    nodejs 核心模块crypto
    es6新特性学习
    nodejs 常用全局包
    ionic+angular+cordova 安卓环境搭建
    谷歌浏览器调试保存到文件
    Linux命令
    Linux中用户管理详解(上)-Linux学习日记
    liunx下忘记root密码的解决方法
    cvCanny的参数
    VC运行时库(/MD、/MT等)
  • 原文地址:https://www.cnblogs.com/chenjx85/p/9631927.html
Copyright © 2011-2022 走看看