在shell脚本中将十进制数转换为十六进制和二进制

时间:2013-09-18 10:49:25

标签: bash shell awk binary hex

我在file.txt

的每一行都有一个十进制数字
    1
    2
    3

我正在尝试(现在已经很长时间)编写一个单行脚本来生成一个输出,其中每一行都有一个带十进制,十六进制和二进制的列。为了简化任务,我们可以说原始数字以字节表示。所以最大值是255。

  1. 我首先尝试将每个数字解码为前缀为0的旁数,以便具有8位模式:

    awk '{print "ibase=10;obase=2;" $1}' $1 | bc | xargs printf "%08d\n"

    其中awk语句中的外部$ 1为file.txt。输出是:

    00000001
    00000010
    00000011
    
  2. 相同的是十六进制,其中一个前置为0

    awk '{printf("0x%02x\n", $1)}' $1

    和以前一样。输出是:

    0x01
    0x02
    0x03
    
  3. 嗯,小数应该只是一个打印:

    1
    2
    3
    
  4. 我想要的是我所拥有的一个班轮:

        1 00000001 0x01
        2 00000001 0x02
    

    所以基本上把1. 2.和3.放在输出的每一行。

    我尝试使用system()在awk中执行bc(和其他命令)但没有成功。还有其他方式。你有什么办法呢?

3 个答案:

答案 0 :(得分:7)

以下单行应该有效:

printf "%s %08d 0x%02x\n" "$1" $(bc <<< "ibase=10;obase=2;$1") "$1"

示例输出:

$ for i in {1..10}; do printf "%s %08d 0x%02x\n" "$i" $(bc <<< "ibase=10;obase=2;$i") "$i"; done
1 00000001 0x01
2 00000010 0x02
3 00000011 0x03
4 00000100 0x04
5 00000101 0x05
6 00000110 0x06
7 00000111 0x07
8 00001000 0x08
9 00001001 0x09
10 00001010 0x0a

答案 1 :(得分:4)

您不需要bc。这是一个仅使用awk的解决方案:

  • 获取bits2str函数available in the manual
  • 添加此最小脚本:

    {
        printf("%s %s %x\n", $1, bits2str($1), $1)
    }
    

这会产生:

$ awk -f awkscr.awk nums 
1 00000001 1
2 00000010 2
3 00000011 3

答案 2 :(得分:4)

所以我搜索了一个简短而优雅的awk二进制转换器。不满意认为这是一个挑战,所以在这里。对于大小有点优化,所以我在下面放了一个可读的版本。

最后的printf指定数字应该有多大。在这种情况下是8位。

这是不好的代码吗?嗯,是的......这是awk :-) 当然不适用于非常庞大的数字。

67个字符长awk代码:

awk '{r="";a=$1;while(a){r=((a%2)?"1":"0")r;a=int(a/2)}printf"%08d\n",r}'

编辑:55个字符的awk代码

awk '{r="";a=$1;while(a){r=a%2r;a=int(a/2)}printf"%08d\n",r}'

可读版本:

awk '{r=""                    # initialize result to empty (not 0)
      a=$1                    # get the number
      while(a!=0){            # as long as number still has a value
        r=((a%2)?"1":"0") r   # prepend the modulos2 to the result
        a=int(a/2)            # shift right (integer division by 2)
      }
      printf "%08d\n",r       # print result with fixed width
     }'

问了一个带有bin和hex的衬垫

awk '{r="";a=$1;while(a){r=a%2r;a=int(a/2)}printf"%08d 0x%02x\n",r,$1}'