IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> JAVA中Integer源码的reverse方法全面细致解析 -> 正文阅读

[数据结构与算法]JAVA中Integer源码的reverse方法全面细致解析

在写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位

第四行同理,这里略

  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2021-10-12 23:43:35  更:2021-10-12 23:44:27 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/6 17:11:22-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码