java中位运算在算法中的应用

2024-08-27 13:36
文章标签 java 算法 应用 运算 中位

本文主要是介绍java中位运算在算法中的应用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1.概念

​ 在Java语言中,提供了7种位运算符,分别是按位与(&)、按位或(|)、按位异或(^)、取反(~)、左移(<<)、带符号右移(>>)和无符号右移(>>>)。

​ 这些运算符当中,仅有~是单目运算符,其他运算符均为双目运算符。

  • ​ 对数值类型数据进行按位操作;1表示true、0表示false。
  • ​ 按位运算表示按每个二进制位(bit)进行计算,其操作数和运算结果都是整型值。
  • ​ 位运算符是对long、int、short、byte和char这5种类型的数据进行运算的,我们不能对double、float和boolean进行位运算操作。

位运算通常比算术运算更快,原因有以下几点:


1.  硬件支持:位运算是直接在硬件层面上执行的,现代CPU有专门的电路来处理位运算,因此执行速度非常快。

2.  节省空间:位运算不需要额外的内存空间,操作直接在寄存器中完成。 

3.  原子性:位运算是原子操作,不会出现像浮点运算那样的精度问题。
4.  并行性:位运算可以同时对多个位进行操作,这使得它们可以充分利用现代CPU的并行处理能力。

 凡是位运算符,都是把值先转换成二进制,再进行后续的处理(因为位运算是针对二进制数进行的一种运算,对于十进制这些数值的位运算来说,会先将其转为二进制,再对其进行位运算,之后将运算结果再转为十进制。)

&:按位与

两个操作数对应位同为1时,结果为1,其余全为0。
(或者是只要有一个操作数为0,结果就为0)。

public class Test {public static void main(String[] args) {System.out.println(5 & 3);//结果为1}
}

将2个操作数和结果都转换为二进制进行比较:
5转换为二进制:0000 0000 0000 0000 0000 0000 0000 0101
3转换为二进制:0000 0000 0000 0000 0000 0000 0000 0011

1按位与运算后:0000 0000 0000 0000 0000 0000 0000 0001 

|:按位或

两个操作数对应位同为0时,结果为0,其余全为1。
(或者是只要有一个操作数为1,结果就为1)。

public class Test {public static void main(String[] args) {System.out.println(5 | 3);//结果为7}
}

5转换为二进制:0000 0000 0000 0000 0000 0000 0000 0101
3转换为二进制:0000 0000 0000 0000 0000 0000 0000 0011

7按位或运算后:0000 0000 0000 0000 0000 0000 0000 0111

~:按位非 

第n位为1,那么按位非的结果是第n位变为0,反之亦然。

public class Test {public static void main(String[] args) {System.out.println(~5);//结果为-6}
}

5转换为二进制:0000 0000 0000 0000 0000 0000 0000 0101

-6按位非运算后:1111 1111 1111 1111 1111 1111 1111 1010

补:有朋友对这里-6怎么算的不太理解,我简单解释一下:

5的2进制表示(假设只用4比特表示,最高比特为符号位)是0101,0101按位取反后是1010。1010是补码,取反(符号位不变)加1后就是原码。取反后是1101,加1后是1110(是10进制的-6),所以~5等于-6。

^:按位异或

public class Test {public static void main(String[] args) {System.out.println(5 ^ 3);//结果为6}
}

5转换为二进制:0000 0000 0000 0000 0000 0000 0000 0101
3转换为二进制:0000 0000 0000 0000 0000 0000 0000 0011

6按位异或运算:0000 0000 0000 0000 0000 0000 0000 0110

<<:左位移运算符

符号位不变,低位补0。移几位补几个0。正数或者负数左移,低位都是用0补。

public class Test {public static void main(String[] args) {System.out.println(5<<2);//运行结果是20}
}

0000 0000 0000 0000 0000 0000 0000 0101 左移2位,低位补0:

0000 0000 0000 0000 0000 0000 0001 0100 换算成10进制为20

>>:右位移运算符

如果值为正,则在高位补0,如果值为负,则在高位补1.

public class Test {public static void main(String[] args) {System.out.println(5>>2);//运行结果是1}
}

0000 0000 0000 0000 0000 0000 0000 0101 右移2位,高位补0

0000 0000 0000 0000 0000 0000 0000 0001

<<<:无符号左移运算符

无符号的意思是将符号位当作数字位看待。
即无论值的正负,都在高位补0.

public class Test {public static void main(String[] args) {System.out.println(5>>>3);//结果是0System.out.println(-5>>>3);//结果是536870911}
}

 5换算成二进制: 0000 0000 0000 0000 0000 0000 0000 0101
-5换算成二进制: 1111 1111 1111 1111 1111 1111 1111 1011

 

5无符号右移3位后结果为0,0的二进制为: 0000 0000 0000 0000 0000 0000 0000 0000 // (用0进行补位)
-5无符号右移3位后的结果 536870911 换算成二进制: 0001 1111 1111 1111 1111 1111 1111 1111 // (用0进行补位)

2.位运算进阶 

右移>>(相当于除2)左边补符号位

左移<<(相当于乘2)右边补0

无符号右移>>>(相当于除2)左边补0

        int n3 = 20;    //二进制(1 0100)int n4=n3>>1;   //10(01010)  (最右右移1,最左补符号位,正数补0负数补1)int n5=n3>>2;   //5(101)int n6=n3>>5;   //0(移完了)int n7=n3>>32;  //32的倍数,不变

​ 注意一个细节:

  1. ​ 首先:位移操作同取反操作一样,并不能改变变量本身的值,所能改变的仅是存储在操作数栈中那个数据的值

    public class Demo {public static void main(String[] args) {int a = 5;System.out.println(a<<2); //20  左移2位结果为20System.out.println(a);  //5 但a的值没有改变}
    }
    

位运算符结合赋值操作 

  • &= 按位与赋值
  •  |= 按位或赋值
  •  ^= 按位异或赋值
  •  >>= 右移赋值
  •  >>>= 无符号右移赋值
  •  <<= 赋值左移

这些操作和 “+=” 一个概念

public class Test {public static void main(String[] args) {int a = 5;a &= 3;System.out.println(a);//结果是1}
}

位运算符常见使用

(1) 公式:m*2^n = m << n

练习:

    @Testpublic void test() {System.out.println("2^3=" + (1 << 3));//2^3=8System.out.println("3*2^3=" + (3 << 3));//3*2^3=24}

法则一:任何数左移(右移)32的倍数位等于该数本身。
法则二:在位移运算m<<n的计算中,若n为正数,则实际移动的位数为n%32,若n为负数,则实际移动的位数为(32+n%32),右移,同理。
左移是乘以2的幂,对应着右移则是除以2的幂。

(2)判断一个数n的奇偶性
         n&1 == 1?”奇数”:”偶数”

为什么与1能判断奇偶?所谓的二进制就是满2进1,那么好了,偶数的最低位肯定是0(恰好满2,对不对?),同理,奇数的最低位肯定是1.int类型的1,前31位都是0,无论是1&0还是0&0结果都是0,那么有区别的就是1的最低位上的1了,若n的二进制最低位是1(奇数)与上1,结果为1,反则结果为0.
练习:

    @Testpublic void test() {int n = 2;int m = 3;System.out.println((n & 1) == 1 ? "奇数" : "偶数"); //偶数System.out.println((m & 1) == 1 ? "奇数" : "偶数"); //奇数}

(3)不用临时变量交换两个数

这个知识点面试的时候有可能会被问到
在int[]数组转置的过程中,是不看到过这样的代码:

public static int[] reverse(int[] nums){int i = 0;int j = nums.length-1;while(j>i){nums[i]= nums[i]^nums[j];nums[j] = nums[j]^nums[i];nums[i] = nums[i]^nums[j];j--;i++;}return nums;
}

连续三次使用异或,并没有临时变量就完成了两个数字交换,怎么实现的呢?

   public void test2() {int n = 2;int m = 3;n = n ^ m;m = m ^ n;  //m = m ^ (n ^ m) => m=nn = n ^ m;  //n = (n ^ m)^[m ^ (n ^ m)] => n=mSystem.out.println(n + ";" + m); //3;2}

常见的计算法则:
① a ^ a =0 (任何数异或本身结果为0)
② a ^ b =b ^ a (交换律)
③ a ^ b ^ c = a ^ (b ^ c) = (a ^ b) ^ c (结合律)
④ 0 ^ a = a (异或0具有保持的特点)
⑤ a ^ b ^ a = b (根据①②④可得)

(4)检测是否为正数 

在补码表示法中,一个整数的正负是由其最高位(符号位)决定的:

•  符号位为0,表示正数或零。
•  符号位为1,表示负数。

boolean isPositive(int n) {return (n << 31) >> 31 == n;
}

 

 

 

这篇关于java中位运算在算法中的应用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1111815

相关文章

Java中流式并行操作parallelStream的原理和使用方法

《Java中流式并行操作parallelStream的原理和使用方法》本文详细介绍了Java中的并行流(parallelStream)的原理、正确使用方法以及在实际业务中的应用案例,并指出在使用并行流... 目录Java中流式并行操作parallelStream0. 问题的产生1. 什么是parallelS

Java中Redisson 的原理深度解析

《Java中Redisson的原理深度解析》Redisson是一个高性能的Redis客户端,它通过将Redis数据结构映射为Java对象和分布式对象,实现了在Java应用中方便地使用Redis,本文... 目录前言一、核心设计理念二、核心架构与通信层1. 基于 Netty 的异步非阻塞通信2. 编解码器三、

SpringBoot基于注解实现数据库字段回填的完整方案

《SpringBoot基于注解实现数据库字段回填的完整方案》这篇文章主要为大家详细介绍了SpringBoot如何基于注解实现数据库字段回填的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以了解... 目录数据库表pom.XMLRelationFieldRelationFieldMapping基础的一些代

一篇文章彻底搞懂macOS如何决定java环境

《一篇文章彻底搞懂macOS如何决定java环境》MacOS作为一个功能强大的操作系统,为开发者提供了丰富的开发工具和框架,下面:本文主要介绍macOS如何决定java环境的相关资料,文中通过代码... 目录方法一:使用 which命令方法二:使用 Java_home工具(Apple 官方推荐)那问题来了,

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

Java AOP面向切面编程的概念和实现方式

《JavaAOP面向切面编程的概念和实现方式》AOP是面向切面编程,通过动态代理将横切关注点(如日志、事务)与核心业务逻辑分离,提升代码复用性和可维护性,本文给大家介绍JavaAOP面向切面编程的概... 目录一、AOP 是什么?二、AOP 的核心概念与实现方式核心概念实现方式三、Spring AOP 的关

详解SpringBoot+Ehcache使用示例

《详解SpringBoot+Ehcache使用示例》本文介绍了SpringBoot中配置Ehcache、自定义get/set方式,并实际使用缓存的过程,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录摘要概念内存与磁盘持久化存储:配置灵活性:编码示例引入依赖:配置ehcache.XML文件:配置

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

Java中的.close()举例详解

《Java中的.close()举例详解》.close()方法只适用于通过window.open()打开的弹出窗口,对于浏览器的主窗口,如果没有得到用户允许是不能关闭的,:本文主要介绍Java中的.... 目录当你遇到以下三种情况时,一定要记得使用 .close():用法作用举例如何判断代码中的 input

深入理解Mysql OnlineDDL的算法

《深入理解MysqlOnlineDDL的算法》本文主要介绍了讲解MysqlOnlineDDL的算法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小... 目录一、Online DDL 是什么?二、Online DDL 的三种主要算法2.1COPY(复制法)