结果

结果

我知道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

相关内容