十六进制转八进制的快捷方法——巧用格式化输入输出

    最近刷题的时候遇到一个基础题,就是将16进制数转为8进制数。咋一看极其简单,用二进制做中介即可,简单规划了一下就开始动手了。

问题描述
  给定n个十六进制正整数,输出它们对应的八进制数。

输入格式
  输入的第一行为一个正整数n (1<=n<=10)。
  接下来n行,每行一个由0~9、大写字母A~F组成的字符串,表示要转换的十六进制正整数,每个十六进制数长度不超过100000。

输出格式
  输出n行,每行为输入对应的八进制正整数。

题目链接:基础练习 十六进制转八进制

 

一、二进制作为中介的解法

因为一个十六进制数为4个二进制数组成,而一个八进制数则由三个二进制数组成,所以二进制就成了链接八进制与十六进制的桥梁。又因为输入数据的规模较为庞大,所以我们得申请大量的内存空间。

我们需要申请的内存空间主要分为三部分:存储输入的n个数组,存放中间值的二进制数组,以及存放转换好的八进制的数组。由于题目要求输出必须在全部输入后,所以我们有这样一条流程:

如果只申请一块内存空间用于存放二进制数,那最少也得申请1.2MB左右的空间,再加上进制转化过程中的各种繁琐操作(如必须由后往前读取比特),还没开写,就能想到产出必定是又长又乱,且还满身补丁的奇葩代码(博主照这个流程完成的也的确乱的一蹋糊涂,没通过后就直接重构了,连修改的欲望都没有了。类似的版本百度一下即可得到)。

 

二、格式化输入输出里的转换说明符

我们知道,在格式化输入输出函数中,请求打印变量的指令取决于变量的类型,即转换说明符指定了如何把数据转换成可显示的形式。除了我们常用的%d %f %c之外,还有一些特定的进制数计数法,如:

%i:有符号十进制数; %u:无符号十进制整数; %o:无符号八进制整数; %x:使用0f的无符号十六进制整数; %X: 使用0F的无符号十六进制整数。

所以,在输入的时候以十六进制输入,输出时以八进制输出,就完成了这个程序的基本框架。

  1. /* 
  2.     printf的格式化进行16进制与8进制的转换  
  3. */  
  4. #include <stdio.h>  
  5.     
  6. int main (void)  
  7. {  
  8.     int x = 0;  
  9.     
  10.     scanf("%x",&x);  
  11.     printf("%x: %%d = %d , %%o = %o\n",x,x,x);  
  12.         
  13.     return 0;  
  14. }  

    输入十六进制 'ABC'

    十进制为A*16^2+B*16^1+c*16^0 = 2748,二进制为1010 1011 1100,转换为八进制为5274,正确。

三、sscanf()

    有了借助于转换说明符的方法也不行,因为在<=100000的输出条件下,哪怕是long long long ….也得溢出。所以不能直接就把输出scanf()到变量里。我们在这儿借助一个数组(字符串),把所有的输入都存起来,再把输入流stdin导入到scanf()里。这里我们借助了sscanf()函数。

    Int sscanf( string str, string fmt, mixed var1, mixed var2 ... );

从一个字符串中读进与指定格式相符的数据

 

四、对字符串的操作

    对于一组长度恐怖的数据进行操作,分解是必须要做的。所以,一次读取多少个数据,又该怎么读,就成了下一个要思考的问题。

    在传统的解法,也即将二进制作为中介的解法中,对十六进制的操作是从最后一位开始的,因此对于十六进制'ABCD',把它分成两部分放入scanf()中,与一次读入的结果是不同的。所以对于数组,从低位到高位是可行的方法。

又因为三位十六进制,即12个bit,恰好可以转化为4个八进制,所以一次读取的数量应该为3的倍数。具体的数目也得依据读入该数据的内存大小来定,也即示例代码中'x'的类型。太大会溢出,太小又不高效。

由此可以看出,当一次读入9个F时,对应的十进制数超过了687亿,远远超过了long long 可以表示的范围,所以,一次读取六个字符就成了最好的方案。

 

五、程序各流程操作的顺序

    如果是从后往前读取十六进制字符的话,那么读取到的字符就是反过来的。而输出是要求正序,所以这样的话就要求把处理后的结果保存起来,再输出。但是再次申请内存空间是没必要的,所以我们得做一些处理,让程序正序处理字符串。

    整个待处理的十六进制字符串可分为两部分,一部分是一次六个字符被统一读入的部分,另一部分是没有凑够六个的部分。如果字符长度对六取模的值为零,那么正序后者倒叙的结果都一样。所以先对开头的一部分单独处理,再用一个循环来处理后面的字符串,就成了最直接的方法。

  1. int x = 0;  
  2. j = arr_len[i] % 6;  
  3. if(j != 0)  
  4. {  
  5.     switch(j)  
  6.     {  
  7.         case 1:  
  8.             sscanf(&arr_16[i][0],"%1x",&x);  
  9.             printf("%o",x);  
  10.             break;  
  11.         case 2:  
  12.             sscanf(&arr_16[i][0],"%2x",&x);  
  13.             printf("%o",x);  
  14.             break;  
  15.         case 3:  
  16.             sscanf(&arr_16[i][0],"%3x",&x);  
  17.             printf("%o",x);  
  18.             break;  
  19.         case 4:  
  20.             sscanf(&arr_16[i][0],"%4x",&x);  
  21.             printf("%o",x);  
  22.             break;  
  23.         case 5:  
  24.             sscanf(&arr_16[i][0],"%5x",&x);  
  25.             printf("%o",x);  
  26.             break;  
  27.     }  
  28. }  
  29.     
  30. for(j ;j<arr_len[i];j += 6)  
  31. //接下来循环处理的部分  

    变量j是规则部分开始的下标,所以一个循环可以搞定剩下的部分。

  32. for(j ;j<arr_len[i];j += 6)  
  33. {     
  34.     sscanf(&arr_16[i][j],"%6x",&x);  
  35.     printf("%08o",x);  
  36. }  
  37.     
  38. printf("\n");  

 

六、特殊十六进制数的处理

    上面的模块以及把整个程序交代的差不多了,不过还是有一处细节要交代一下。我们来看一组测试用例:

    理论上,六位十六进制数可以转化成八个八进制数。但在实际操作中,有时读取到的十六进制数太小,以至于不到八位。如果读取的数是单独的十六进制数那当然没问题,但是如果读取的是一个长十六进制的一部分,那就出错了。所以在输出时得把前导零带上。但是十六进制最开始的几个字符的前导零得去掉,所以要进行一次判断。

  1. for(j ;j<arr_len[i];j += 6)  
  2. {     
  3.     sscanf(&arr_16[i][j],"%6x",&x);  
  4.     if(j != 0)  
  5.         printf("%08o",x);  
  6.     else  
  7.         printf("%8o",x);  
  8. }  

 

七、效率比较与源码

    我在csdn上找了一份该题的另一份源码,这是借助二进制的一个版本。提交后,各情况如下:

    新提交的为二进制的版本。可以看到二进制的版本用空间换取了时间,时间消耗极少。而sscanf()版本由于调用了标准库,所以消耗时间较多。

 

    本题源码地址:十六进制转八进制源码

posted on 2018-09-25 22:51  Magic激流  阅读(6353)  评论(0编辑  收藏  举报

导航