shell 通过对一个算术表达式求值并替换结果来执行算术扩展。表达式可以像在双引号中一样来进行处理,并且可以嵌套。
求值算术表达式有以下两种格式:
# $[ expression ]
# $(( expression ))
应用举例
在某些场景特别方便,可以免去写程序的烦琐,如验证某些运算。
下面是验证《深入理解计算系统》练习题2.12的场景
表达式 ~0 将生成一个全1的掩码,不管机器的字大小是多少,可移植。
$ printf "%x\n" $(( ~0 ))
ffffffffffffffff
$ printf "%#x\n" $(( ~0 ))
0xffffffffffffffff
上面的测试显示,shell中,0按位取反后的值是64位的。
shell的printf命令前导字符打印:《shell脚本学习指南》表7-4:printf的标志中描述了格式参数中"#“号的意义,”#“可以用以输出前导"0x”(16进制)、“0”(8进制).
x & 0xFF 生成一个由x的最低有效字节组成的值
$ printf "%#x\n" $(( 0x89ABCDEF & 0xFF ))
0xef
$ printf "%#.8x\n" $(( 0x89ABCDEF & 0xFF ))
0x000000ef
验证
以下x = 0x87654321
A.x的最低有效字节,其他位均置为0
$ printf "%#.8x\n" $(( 0x87654321 & 0xFF ))
0x00000021
B.除了x的最低有效字节外,其他的位置都取补,最低有效字节保持不变。
$ printf "%#x" $(( 0x87654321 ^ ~0xff))
0xffffffff789abc21
上面因为~0xff会生成64位的掩码,所以结果有些不符合预期,但后32位是符合预期的。
C.x的最低有效字节设置成全1,其他字节都保持不变。
$ printf "%#x" $(( 0x87654321 | 0xff ))
0x876543ff
root@petalinux:~# printf "%#x\n" $(( (( 0x1 << 31)-1) ))
0x7fffffff
root@petalinux:~# hex=0x8AABCDEF
root@petalinux:~#
root@petalinux:~# printf "%#x\n" $(( hex & (( 0x1 << 31)-1) ))
0xaabcdef
root@petalinux:~#