我在file.txt
:
1
2
3
我正在尝试(现在已经很长时间)编写一个单行脚本来生成一个输出,其中每一行都有一个带十进制,十六进制和二进制的列。为了简化任务,我们可以说原始数字以字节表示。所以最大值是255。
我首先尝试将每个数字解码为前缀为0的旁数,以便具有8位模式:
awk '{print "ibase=10;obase=2;" $1}' $1 | bc | xargs printf "%08d\n"
其中awk语句中的外部$ 1为file.txt
。输出是:
00000001
00000010
00000011
相同的是十六进制,其中一个前置为0
awk '{printf("0x%02x\n", $1)}' $1
和以前一样。输出是:
0x01
0x02
0x03
嗯,小数应该只是一个打印:
1
2
3
我想要的是我所拥有的一个班轮:
1 00000001 0x01
2 00000001 0x02
所以基本上把1. 2.和3.放在输出的每一行。
我尝试使用system()在awk中执行bc(和其他命令)但没有成功。还有其他方式。你有什么办法呢?
答案 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}'