Java位运算经典实例

一 源码、反码、补码

正数的源码、反码、补码相同,例如5:

           5的源码:101

           5的反码:101

           5的补码:101

负数的源码、反码、补码不同,例如-5:

           -5的源码:10000101

           -5的反码:111111010 (取反操作)

           -5的补码:111111011 (补码加1操作)

计算机所有数据都以补码存储和运算。

二 位操作

      位操作包含&,|,!分别表示与,或非。

    eg:

               5 & 4 = 101 & 100 = 100(按位取“与”,1 & 1 = 1,0 & 1 = 0)

               5 | 4 = 101 | 100 = 101  (按位取“或”,1 | 0 = 1, 0 | 0 = 0)

                    !5    = ! 101 = 010  (按位取“反”,!1  = 0, !0  = 1)

三 位运算实例

          eg:给定一个有符号整数X(32位),写一个方法,检查这个数是否是4的N次方,N是非负整数。

下面是用Java解答这道题:

public class Test {
    public static void main(String[] args) {
        for(int i = -64; i < 400; i+=1) {
            if(isPowerOfFour5(i))
            System.out.println("test "+ i + " is power of four!");
        }
    }

    public static boolean isPowerOfFour1(int x) {
        if(x == 0) return false;
        while ((x % 4) == 0) {
            x /= 4;
        }
        return x == 1;
    }
   
    public static boolean isPowerOfFour2(int x) {
        if(x == 0) return false;
        while ((x % 4) == 0) {
            x >>= 2;
        }
        return x == 1;
    }
   
    public static boolean isPowerOfFour3(int x) {
         double n = Math.log(x) / Math.log(4);
         if(n -(int)n != 0) return false;
         return n >= 0;
    }
   
    public static boolean isPowerOfFour4(int x) {
        if(x == 0) return false;
       int y = (int) Math.sqrt(x);
        if(y * y == x)
            return ((y & (y-1)) == 0);
        return false;
    }
   
    public static boolean isPowerOfFour5(int x) {
        if(x == 0) return false;
        return (x & (x - 1)) == 0 && (x & 0xAAAAAAAA) == 0;
    }
}

稍微解释一下,第一个方法的算法复杂度是log4X,原理是,凡是4的N次方的数(N是大于等于0的正整数),则对X一直取余,最终结果肯定等于1,比如4 * 4 * 4 = 64。

第二个方法和第一个方法原理一样,>>2相当于/4。

第三个方法利用公式:4N = X  => N = log4X = logX / log4,所以只需要判断N大于等于0且为整数即可(如果n-(int)n != 0表明n不是整数)。

第四个方法的原理如下:

40     41   42    43   …  4n

(22)0 (22)1 (22)2 (22)3 … (22)n

则y = Math.sqrt(x)等于:

20   21    22    23   …  2n

所以我们只需要判断y是整数,且y是2的n次方,n大于等于0且为整数:

判断一个整数y是不是2的n次方,只需要判断y&(y-1)等不等于0即可。

第五种方法的原理如下:

40                        1

41                   100

42              10000

43         1000000

 

4n     100000000…

可以发现,所有以4为底的N次方的数的2进制都只有1位为1,且为1的这一位在奇数位上。

因此,我们首先判断X的2进制只有一位,且该位不在偶数位上,还要注意0必须除外,因为0&任何数都等于0。

前面说过,x & (x - 1) == 0则表明x是2的n次方,我们应该知道任何2进制只有一位为1的数都可以表示为2的n次方。

既然已经确定x的2进制数只有一位为1,那么(x & 0xAAAAAAAA) == 0表明为1的这一位在奇数位上,因为0xAAAAAAAA = 1010 1010 1010 1010

 

时间: 2024-08-01 22:47:33

Java位运算经典实例的相关文章

Java千问:Java位运算经典应用(一)

很多人认为位运算在实际开发过程中并没什么用,学习位运算也只是为了应付面试.这种想法是错误的,接下来我们就通过几篇连载文章介绍一下位运算在实际开发过程中的几个经典应用实例.如果对位运算规则掌握还不是很熟练,可以先阅读<Java千问:Java语言位运算符详解>.这篇文章不仅详细讲解了Java位运算的基本规则和一些常用的运算定律,同时还在文中提到了一些常用的位运算实际应用,比如可以用位运算操作的方式快速把某个变量所在的内存单元清零,或者位运算的方式实现某个变量快速倍增等等.但文中所这提到的这几个实际

Java千问:Java位运算经典应用(二)

接上篇 三.不借助中间变量交换两个变量的值 通常情况下,我们要交换两个变量的值都按如下步骤操作: 这种操作方式不难理解,实现交换变量值的关键点就在于中间变量c.而现在的题目要求是不借助中间变量来交换a和b的值.如果不使用位运算的方式,同样可以做到不借助中间变量交换两个变量的值,其实现过程如下. 为了讲解方便,我们把最初a与b的值称之为原始a和原始b,3行代码就是3步操作:第1步:把原始a与原始b相加的和存储到变量a中,变量b的值暂时没有发生变化.第2步:用这个和减去原始b,再赋值到变量b中,经过

java位运算应用

位移动运算符: <<表示左移, 左移一位表示原来的值乘2. 例如:3 <<2(3为int型) 1)把3转换为二进制数字0000 0000 0000 0000 0000 0000 0000 0011, 2)把该数字高位(左侧)的两个零移出,其他的数字都朝左平移2位, 3)在低位(右侧)的两个空位补零.则得到的最终结果是0000 0000 0000 0000 0000 0000 0000 1100, 转换为十进制是12. 同理,>>表示右移. 右移一位表示除2. 位运算:

Java位运算总结:位运算用途广泛《转》

前天几天研究了下JDK的Collection接口,本来准备接着研究Map接口,可是一查看HashMap类源码傻眼咯,到处是位运算实现,所以我觉得还是有必要先补补位运算知识,不然代码看起来有点费力.今天系统研究了下,现记录如下. 首先要明白一个概念,Java位运算是针对于整型数据类型的二进制进行的移位操作.主要包括位与.位或.位非,有符号左移.有符号右移,无符号右移等等.需要注意一点的是,不存在无符号左移<<<运算符.根据位运算的概念规定,我们首先需要弄明白两个问题,java有哪些数据类型

Java 位运算超全面总结

1.原码.反码.补码 关于原码.反码.补码的相关知识作者不打算在这里长篇大论,相关知识已有别的大佬总结很好了,还请老铁自行 Google,不过有篇知乎回答是作者学编程以来见过对相关知识最通俗易懂,生动简洁的解释:对原码.反码.补码最通俗易懂,生动简洁的解释,墙裂建议大家先看完这篇科普文章.在继续讨论之前你要先明白一点:整数在计算机内部都是以补码形式存储的. 2.Java 位运算概览 OK 都看到这儿了那我就假定你已经掌握了原码.反码.补码相关知识(虽然上面那段几乎啥也没讲,纯凑字数) 不废话了.

Java 位运算

一,Java 位运算 1.表示方法: 在Java语言中,二进制数使用补码表示,最高位为符号位,正数的符号位为0,负数为1.补码的表示需要满足如下要求. (l)正数的最高位为0,其余各位代表数值本身(二进制数). (2)对于负数,通过对该数绝对值的补码按位取反,再对整个数加1. 2.位运算符 位运算表达式由操作数和位运算符组成,实现对整数类型的二进制数进行位运算.位运算符可以分为逻辑运算符(包括~.&.|和^)及移位运算符(包括>>.<<和>>>). 1)左

(转)java位运算

转自:http://aijuans.iteye.com/blog/1850655 Java 位运算(移位.位与.或.异或.非) public class Test { public static void main(String[] args) { // 1.左移( << ) // 0000 0000 0000 0000 0000 0000 0000 0101 然后左移2位后,低位补0:// // 0000 0000 0000 0000 0000 0000 0001 0100 换算成10进制为

Java 位运算(移位、位与、或、异或、非)与逻辑运算

java 位运算包括:左移( << ).右移( >> ) .无符号右移( >>> ) .位与( & ) .位或( | ).位非( ~ ).位异或( ^ ),除了位非( ~ )是一元操作符外,其它的都是二元操作符. 逻辑运算符&.&&.|.||: 一.逻辑&与短路&&的区别 总的来说区别是体现在,只有这两个运算符的左边为false的时候会有区别,看如下代码 1.逻辑&的运算 boolean a = tr

Java 位运算2-LeetCode 201 Bitwise AND of Numbers Range

在Java位运算总结-leetcode题目博文中总结了Java提供的按位运算操作符,今天又碰到LeetCode中一道按位操作的题目 Given a range [m, n] where 0 <= m <= n <= 2147483647, return the bitwise AND of all numbers in this range, inclusive. For example, given the range [5, 7], you should return 4. 题意: