您的位置:首页 > 编程语言 > Java开发

java使用移位运算进行进制转化

2017-03-22 09:44 429 查看
最近在练习进制转换算法,学习了一种使用移位运算进行进制的方法分享给大家。

16进制转换8进制

问题描述:
输入格式
输入的第一行为一个正整数n (1<=n<=10)。
接下来n行,每行一个由0~9、大写字母A~F组成的字符串,
表示要转换的十六进制正整数,每个十六进制数长度不超过100000。
输出格式
输出n行,每行为输入对应的八进制正整数。
注意
输入的十六进制数不会有前导0,比如012A。输出的八进制数也不能有前导0。
样例输入
239123ABC
样例
71
4435274


我一开始的解法,1个16进制数可以转化为4个2进制数,再通过每3个二进制数可以转化为1个8进制数。不过有一点需要注意转化成8进制数要从2进制字符串后面开始,到最前面时需要判断还剩下几位,不够3位要补0。

package 十六进制转八进制;

import java.text.DecimalFormat;
import java.util.Scanner;
import java.util.Stack;
public class Main {
static Scanner scan = new Scanner(System.in);
static DecimalFormat decimalFormat=new DecimalFormat("0000");
static DecimalFormat d2=new DecimalFormat("000");
public static void change(String str){

String s = "";

int top = 0;
char[] c = str.toCharArray();
for (int k = 0; k < c.length; k++) {
if (c[k] >= '0' && c[k] <= '9') {
s+=decimalFormat.format(Integer.parseInt(Integer.toBinaryString(c[k]-48)));
}else if(c[k] >= 'A' && c[k] <= 'F'){
s+=decimalFormat.format(Integer.parseInt(Integer.toBinaryString(c[k]-55)));
}
}
int num = s.length();
if(num%3==1)
s="00"+s;
else if(num%3==2)
s="0"+s;
if(s.substring(0, 3).equals("000")){
s=s.substring(3);
}
for (int i = 0; i < s.length(); i=i+3) {

String xStr = s.substring(i,i+3);
System.out.print(Integer.parseInt(xStr.substring(0,1))*4+
Integer.parseInt(xStr.substring(1,2))*2+
Integer.parseInt(xStr.substring(2,3))*1);
}

}
public static void main(String[] args) {
int n = Integer.parseInt(scan.nextLine());
String[] strs = new String
;
for (int i = 0; i < strs.length; i++) {
strs[i] = scan.nextLine();
}
for (int i = 0; i < strs.length; i++) {
change(strs[i]);
System.out.println();
}
}
}


使用题目给的测试可以通过,我自己设置的测试数据也可以通过,然后去蓝桥杯测试平台上跑结果炸了!去看测试数据,10个16进制数,最长的10万位,估计是溢出了。去调试发现是在2进制转化为8进制的时候出错了,for循环次数太多了。for里还有数值运算比较费时,改了好久测试总是通不过。去百度发现一种运用java移位运算进行转化的,写了试了一下发现可以通过了。

使用java移位运算符进行转化

import java.util.Scanner;
public class Main {
public static void main(String[] args) {
new Main().systemScanner();
}
public void systemScanner() {
Scanner jin = new Scanner(System.in);
while (jin.hasNext()) {
int length = jin.nextInt();
for (int i = 0; i < length; i++){
String strTmp=jin.next();
tranform(strTmp.toCharArray(), strTmp.length());
}
}
}
/*
* 3位16进制等价于4位8进制
*/
int[] stack=new int[40000];
public void tranform(char[] str, int length) {
char[] buff = new char[4];
int top = -1;
for (int i = length - 1; i >= 0; i -= 3) {
int sum = 0;
for (int j = 0; j < 3 && i - j >= 0; j++) {// i-j>=0防止不够三个的情况
int tmp = str[i - j] >= '0' && str[i - j] <= '9' ? str[i - j] - '0': str[i - j] - 'A' + 10;//区分是数字,还是字符,进行对应转换
sum+=(tmp<<(4*j));//这句很重要,通过这句就可以从16变成10进制了。
}
stack[++top]=sum;//sum的结果是16进制转化10进制的结果,每3个16进制变成10进制,再变8进制
}
while(stack[top]==0){//排除前导为0的判断
top--;
}
for(int i=top;i>=0;i--){
String str1=Integer.toOctalString(stack[i]);//从10进制转化成8进制
if(i!=top&&str1.length()<4){
//不是最左边的一个,就不用去掉前导0,而默认是去掉0的,所以要进行补会
for(int y=0;y<4-str1.length();y++)
System.out.print("0");
}
System.out.print(str1);
}
System.out.println();
}
}


这种解法的思路是:1位16进制可以代表4位2进制, 1位8进制可以代表3位二进制,得出3位16进制求和入栈输出表示4位8进制,然后出栈输出。由16进制转化为10进制的时候,使用 << 使16进制数转化为8进制。

例子:

0x11 << 4 => 17;

/*
11的二进制编码是 1011,个位是1 编码是0001,十位是1 编码也是0001。
根据位数进行移位运算,个位左移(4*0)位还是0001,十位左移(4*1)位变成00010000
*/

0001 << 4*1 = 00010000; 0001 << 4*0 = 0001;

/*
整个数就变成了00010001 也就是10001 转换为10进制是为17。这里可以看到一个16进
制数通过按位数分别左移4*(n-1)位就可以转化为10进制数了。具体的原因是每一个整
数左移4位等于这数的二进制表示后面多了4个0,相当于2进制乘法乘了一个二进制10000()。
*/
1*16^1+1*16^0 = 17;
/*
10000转化为10进制为16也就是每位乘了16^(n-1)进而把一个16进制数转化为10进制数。
*/


这种算法的效率非常高,运算十分较少。可以通过蓝桥杯测试系统的变态数据。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: