Reverse bits of a given 32 bits unsigned integer. For example, given input 43261596 (represented in binary as 00000010100101000001111010011100), return 964176192 (represented in binary as 00111001011110000010100101000000). Follow up: If this function is called many times, how would you optimize it?
这里涉及了十进制和二进制之间转化这个知识点。在数学中,咱们都是采用十进制进行计算,通俗说就是逢十进一。可是计算机没有那么多类型的信号来对应是十进制数中的十个数字,所以采用了二进制来进行计算。面试
二进制,也就是所谓的逢二进一,每一位上只有两个数字,分别是0和1。每个十进制都有一个惟一的二进制数做为对应。咱们先来看一下十进制数的计算,好比945
等价于9*100+4*10+5*1
,也就是9*10^2 + 4*10^1 + 5*10^0
。同理咱们知道,二进制数011
表明0*2^2 + 1*2^1 + 1*2^0
也就是十进制的3
。这样,咱们知道如何将二进制数转化为十进制数。缓存
那么计算机是怎么进行计算的呢?计算机首先将十进制数转化为二进制数,再将二进制数根据需求进行计算。假设计算机须要计算3+6
,那么转化为二进制就是11+110
,那么计算机的计算流程以下。微信
1.对齐 在左侧补零位使的两个数字位数相同,所以就是011+110
2.逢二进一 相似于十进制数的逢十进一,所以计算的结果为1001
,转化为十进制数就是9
基于二进制的位计算上,除去加减乘除,计算机还定义了一些其余的位运算符this
>>
将二进制数右移,左边根据数字的正负形补0或者补1,正数补0,负数补1。
<<
将二进制数左移,右边补0
>>>
将二进制数右移,左边补0
在这些知识的基础上,咱们再来看这道题目。spa
将比特位逆转过来也就是将十进制数转化为二进制数,再从右往左得到每一位上的值,再将这个值添加至结果值中。设计
public int reverseBits(int n) { //得到最后一位的值 int mask = 1; int result = 0; for(int i=0 ; i<32 ; i++){ result <<= 1; result |= (n & mask); n >>= 1; } return result; }
为了将比特位逆转,意味着咱们至少须要32次循环才能将整个遍历完成。那么若是这个逆转比特位方法会被调用屡次的话,咱们能够使用一个Map来缓存已经遍历过的状况。
可是,32位的重复值每每不多,所以咱们能够将整数分解为多个片断存入缓存中,只要遇到重复的片断,就能够将已经获得的结果返回给调用方。根据分治思想,逆转32个比特位等价于分别将每8个比特位进行逆转。所以咱们能够将长度为32的unsigned int拆解成4个长度为8的byte。每8个byte对应的值能够做为片断存在缓存中,只要遇到重复的片断,就能够直接返回。code
Map<Byte, Integer> cache = new HashMap<Byte, Integer>(); public int reverseBits2(int n){ byte[] bytes = new byte[4]; for(int i = 0 ; i<4 ; i++){ // 得到8位并转换为1个byte bytes[i] = (byte) ((n>>>(i*8)) & 0xff); } int result = 0; for(int i = 0 ; i<4 ; i++){ result += reverseByte(bytes[i]); if(i<3) result<<=8; } return result; } public int reverseByte(byte b){ Integer value = cache.get(b); if(value != null) return value; value = 0; for(int i = 0 ; i<8 ; i++){ value += ((b>>>i) & 1); if(i<7) value <<= 1; } cache.put(b, value); return value; }
想要了解更多开发技术,面试教程以及互联网公司内推,欢迎关注个人微信公众号!将会不按期的发放福利哦~教程