Je voudrais faire l'opération suivante dans mon script:
1 - ((m - 20) / 34)
Je voudrais assigner le résultat de cette opération à une autre variable. Je veux que mon script utilise des maths en virgule flottante. Par exemple, pour m = 34:
results = 1 - ((34 - 20) / 34) == 0.588
Vous pouvez utiliser la calculatrice bc
. Il effectuera des calculs de précision arbitraires à l'aide de nombres décimaux (pas de virgule flottante binaire) si vous définissez increease scale
à partir de la valeur par défaut 0:
$ m=34
$ bc <<< "scale = 10; 1 - (($m - 20) / 34)"
.5882352942
L'option -l
chargera la bibliothèque mathématique standard et définira l'échelle sur 20:
$ bc -l <<< "1 - (($m - 20) / 34)"
.58823529411764705883
Vous pouvez ensuite utiliser printf pour formater la sortie, si vous le souhaitez:
printf "%.3f\n" "$(bc -l ...)"
Bash ne fait pas de maths en virgule flottante. Vous pouvez utiliser awk ou bc pour gérer cela. Voici un exemple awk:
$ m=34; awk -v m=$m 'BEGIN { print 1 - ((m - 20) / 34) }'
0.588235
Pour affecter la sortie à une variable:
var=$(awk -v m=$m 'BEGIN { print 1 - ((m - 20) / 34) }')
Enseigner bash par exemple division entière avec résultats en virgule flottante:
#!/bin/bash
div () # Arguments: dividend and divisor
{
if [ $2 -eq 0 ]; then echo division by 0; exit; fi
local p=12 # precision
local c=${c:-0} # precision counter
local d=. # decimal separator
local r=$(($1/$2)); echo -n $r # result of division
local m=$(($r*$2))
[ $c -eq 0 ] && [ $m -ne $1 ] && echo -n $d
[ $1 -eq $m ] || [ $c -eq $p ] && return
local e=$(($1-$m))
let c=c+1
div $(($e*10)) $2
}
result=$(div 1080 633) # write to variable
echo $result
result=$(div 7 34)
echo $result
result=$(div 8 32)
echo $result
result=$(div 246891510 2)
echo $result
result=$(div 5000000 177)
echo $result
Sortie:
1.706161137440
0.205882352941
0.25
123445755
28248.587570621468
echo $a/$b|bc -l
donne le résultat.
Exemple:
read a b
echo $a/$b|bc -l
Entrez a & b comme valeur 10 3, vous obtenez 3.3333333333
Si vous voulez stocker la valeur dans une autre variable, utilisez le code
read a b
c=`echo $a/$b|bc -l`
echo $c
Il donne également le même résultat que ci-dessus . Essayez-le ...
Je sais que c'est un vieux fil, mais cela semblait être un projet amusant à aborder sans utiliser bc
ou invoquer la récursion. Je suis sûr que cela peut être amélioré, mais cela a maximisé mes compétences.
numerator=5
denominator=7 # - 0 -> returns "undef"
decimal_places=4 # - 0 -> same as echo $(( $numerator / $denominator ))
_result_sign=""
let _dp_exp=10**decimal_places
if [ $denominator -eq 0 ]; then _div_result_int_large=0; else let _div_result_int_large=$((numerator * _dp_exp / denominator)); fi
if [ $_div_result_int_large -lt 0 ]; then let _div_result_int_large=$(( _div_result_int_large * -1 )); _result_sign="-"; fi
let _div_result_int=$((_div_result_int_large / _dp_exp))
let _div_result_mant=$((_div_result_int_large - _div_result_int * _dp_exp))
let _dp_lzeros=$((decimal_places - ${#_div_result_mant}))
printf -v _div_result_mant_padded "%.${_dp_lzeros}d$_div_result_mant"
div_result="$_result_sign$_div_result_int"
if [ $decimal_places -gt 0 ]; then div_result="$_result_sign$_div_result_int.$_div_result_mant_padded"; fi
if [ $denominator -eq 0 ]; then div_result="undef"; fi
echo $div_result
Exemple de sortie:
numerator=5
denominator=7
decimal_places=5
-> 0.71428
numerator=250
denominator=13
decimal_places=0
-> 19
numerator=-5
denominator=6
decimal_places=2
-> -0.83
numerator=3
denominator=0 # - uh-oh
decimal_places=2 # - can be anything, in this case
-> undef