我知道BC:
$> var="304"
$> echo "obase=2; $var" | bc
100110000
可以在 shell 中完成(无需外部调用)吗?
这个问题: shell脚本中的二进制转十六进制和十进制 询问如何从二进制转换为二进制数。
那里的答案要么使用 xxd 处理二进制字节(与二进制数相反,即:以 2 为基数的数字),要么使用其他一些外部工具。所以,不,这个问题不是这个问题的重复。
答案1
perl
是完成这项工作的好工具:
$ var="304"
$ perl -e 'printf "%b\n",'$var
100110000
答案2
在bash中:
toBinary(){
local n bits sign=''
(($1<0)) && sign=-
for (( n=$sign$1 ; n>0 ; n >>= 1 )); do bits=$((n&1))$bits; done
printf "%s\n" "$sign${bits-0}"
}
使用:
$> toBinary 304
100110000
或者更多 POSIX_ly:
toBinaryPOSIX(){
n=$(($1))
bits=""
sign=""
if [ "$n" -lt 0 ]; then
sign=- n=$((-n))
fi
while [ "$n" -gt 0 ]; do
bits="$(( n&1 ))$bits";
: $(( n >>= 1 ))
done
printf "%s\n" "$sign${bits:-0}"
}
使用:
$> toBinaryPOSIX 304
100110000
如果值为十六进制:
$> toBinaryPOSIX 0x63
1100011
答案3
以下解决方案仅使用内置函数将数字转换为二进制形式:
#! /bin/sh
E=[24680] # even nums
O=[13579] # odd nums
lookuptbl() {
case $1 in
$E[01]* ) echo 0 ;; $E[23]* ) echo 1 ;;
$E[45]* ) echo 2 ;; $E[67]* ) echo 3 ;;
$E[89]* ) echo 4 ;; $O[01]* ) echo 5 ;;
$O[23]* ) echo 6 ;; $O[45]* ) echo 7 ;;
$O[67]* ) echo 8 ;; $O[89]* ) echo 9 ;;
esac
}
divby2() {
set -- "0$1"
while case $1 in ? ) break ;; esac; do
set -- "${1#?}" "${2-}$(lookuptbl "$1")"
done
echo $2
}
dec2bin() {
case $1 in *[!0]* ) :;; * ) echo 0; return ;; esac
while :
do
case $1 in *[!0]* ) :;; * ) break;; esac
case $1 in
*$E ) set -- "$(divby2 "$1")" "0${2-}" ;;
*$O ) set -- "$(divby2 "$1")" "1${2-}" ;;
esac
done
echo "$2"
}
# And then... (assuming pure number given from command line)
num=${1:-304}
dec2bin "$num"
结果
100110000
答案4
您可以使用 POSIX awk标准库:
$ awklib 'BEGIN {print base_conv(304, 10, 2)}'
100110000
或者:
$ awklib 'BEGIN {print base_conv(ARGV[1], 10, 2)}' 304
100110000