ICode9

精准搜索请尝试: 精确搜索
首页 > 编程语言> 文章详细

JAVA中Integer源码的reverse方法全面细致解析

2021-10-11 23:58:16  阅读:206  来源: 互联网

标签:运算 JAVA reverse int 偶数 M4 源码 M1 翻转


在写leetcode网站的“”190. 颠倒二进制位”算法题时发现了这个方法的源码,于是有了如下研究成果。

大致思想原理

若要翻转一个二进制串,可以将其均分成左右两部分,再对划分出的左右部分各自进行划分左右,直至每部分只有一个元素
(对每部分递归执行翻转操作)
然后将左半部分拼在右半部分的后面,即完成了翻转
(例如12,左半放右半后面变为21就是翻转;1234,先划分左右,到各区域仅1个元素后开始返回,第一次返回2143,第二次返回4321,翻转完成)
由于左右两部分的计算方式是相似的,利用位掩码和位移运算,我们可以自底向上地完成这一分治流程。

对于递归的最底层(一位一组),我们需要交换所有奇偶位:
1、取出所有奇数位和偶数位;
2、将奇数位移到偶数位上,偶数位移到奇数位上。
类似地,对于倒数第二层(每两位分一组),按组号取出所有奇数组和偶数组,然后将奇数组移到偶数组上,偶数组移到奇数组上。以此类推。
需要注意的是,在某些语言(如 Java)中,没有无符号整数类型,因此对 n 的右移操作应使用逻辑右移。

更进一步解释

翻转原理:以1234为例子:翻转过来就是把最左侧的数和最右的数交换,然后向内进一层再重复交换直至内层只剩一个数或没有数。
而1234翻转得到4321,只看区域变化就是原先的左半部分和右半分交换了位置,先保留这条性质,如果只是这么交换得到的是3412,此时和正确的翻转有差异,因为左半区域和右半区域各自区域内没有进行左右交换。而把这个顺序逆过来,先对总区域划分到左右区域各只有一个元素,然后交换,再逐层往上交换,就是翻转原理。

代码节选(本质不变,为便于描述进行了轻微修改):

public class Solution {
    private static final int M1 = 0x55555555; // 二进制为01010101010101010101010101010101
    private static final int M2 = 0x33333333; // 00110011001100110011001100110011
    private static final int M4 = 0x0f0f0f0f; // 00001111000011110000111100001111
    private static final int M8 = 0x00ff00ff; // 00000000111111110000000011111111

    public int reverseBits(int n) {
        n = n >>> 1 & M1 | (n & M1) << 1;
        n = n >>> 2 & M2 | (n & M2) << 2;
        n = n >>> 4 & M4 | (n & M4) << 4;
        n = n >>> 8 & M8 | (n & M8) << 8;
        return n >>> 16 | n << 16;
    }
}

代码原理:

各行原理相同,以 n = n >>> 1 & M1 | (n & M1) << 1; 为例:执行顺序为
含义为将 n >>> 1 & M1 的结果和 (n & M1) << 1 的结果进行 | 运算,然后赋值给 n
n >>> 1 & M1 意为将 n >>> 1 的结果和 M1 做 & 运算,| 运算右侧的同理

第一行代码的效果:每两位互换
(n & M1) << 1:取奇位数并左移一位
(因为M1只有奇数位才使1,而&运算规定只有两1的才能为1,而偶数为全为0会使二者运算结果只保留奇数位的原值,偶数位全为0)
n = n >>> 1 & M1:右移一位再取奇位数
(相当于把(n & M1) << 1没取到的偶数位都取到了)
二者做 | 运算,假设原数据为123456(为方便描述瞎取的,6为最低位),则左侧结果为204060,右侧结果为010305,二者做或运算(含1为1),正好左侧为0的位置右侧处不为0,运算完正好是两两换位,肉眼也能看出。

第二行代码:| 运算符左半部分效果:每四位,后两位移到前两位。右半部分效果:每四位,前两位移到后面两位
(n & M2) << 2:4位之中取前两位左移两位(与第一行的同理)
(i >>> 2) & 0x33333333 : 右移两位取前两位(与第一行的同理)

第三行代码:原理同第二行,每四位交换了一下
(n & M4) << 4:取前4位左移4位
n = n >>> 4 & M4:右移4位取前4位

第四行同理,这里略

标签:运算,JAVA,reverse,int,偶数,M4,源码,M1,翻转
来源: https://blog.csdn.net/m0_52937388/article/details/120714413

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有