logo Debian Debian Debian-France Debian-Facile Debian-fr.org Forum-Debian.fr Debian ? Communautés logo inclusivité

Debian-facile

Bienvenue sur Debian-Facile, site d'aide pour les nouveaux utilisateurs de Debian.

Vous n'êtes pas identifié(e).


L'icône rouge permet de télécharger chaque page du wiki visitée au format PDF et la grise au format ODT → ODT PDF Export

Différences

Ci-dessous, les différences entre deux révisions de la page.

Lien vers cette vue comparative

Les deux révisions précédentes Révision précédente
Prochaine révision
Révision précédente
Prochaine révision Les deux révisions suivantes
doc:programmation:shells:page-man-bash-iii-les-operateurs-de-comparaison-numerique [15/02/2023 21:52]
ubub [Comparaison numérique avec ((]
doc:programmation:shells:page-man-bash-iii-les-operateurs-de-comparaison-numerique [20/02/2023 21:31]
agp91 [Conclusion sur les opérateurs lexicographiques et les opérateurs de comparaison numérique]
Ligne 24: Ligne 24:
 ===== Introduction ===== ===== Introduction =====
  
-En interne, bash dispose de plusieurs commandes pour réaliser des comparaisons numériques:​ +<note> 
-  *La commande composée **<nowiki>((</​nowiki>​** et la commande ​**let**. +Dans la page du manuel de bash, **les opérateurs des commandes ​de test** sont nommées __**primitives**__
-  *Les commandes **[** et **test**. +</note>
-  *Et la commande composée **<​nowiki>​[[</nowiki>**.+
  
-Les commandes **[** et **test** sont disponibles dans leurs versions externe : **/​usr/​bin/​[** et **/​usr/​bin/​test**.+Bash dispose de plusieurs ​commandes ​pour **réaliser des comparaisons numérique**.
  
 +  * Les commandes de test :
 +    * Les commandes internes **[** et **test**.
 +    * Et la commande composée **<​nowiki>​[[</​nowiki>​**.
 +  * Les commandes d'​évaluation numériques :
 +    * La commande composée **<​nowiki>​((</​nowiki>​** et la commande interne **let**.
 +
 +<​note>​
 +  * Les commandes **<​nowiki>​((</​nowiki>​** et **let** sont équivalentes.
 +  * Les commandes **[** et **test** sont équivalentes.
 +  * Les commandes **[** et **test** sont disponibles dans leurs versions externe : **/​usr/​bin/​[** et **/​usr/​bin/​test**.
 +    * Elles ont toutes les deux la même page de manuel (**man [** ou **man test**).
 +  * Les commandes internes disposent de primitive que n'ont pas les commandes externes.
 +</​note>​
 +
 +<​note>​
 __Rappels :__ __Rappels :__
-  * Une commande de test renvoie le code de retour true (0) lorsque le test réussi et false (1) lorsqu'​il échoue. ​ 
-  * Le code retour d'une commande est mémorisé dans le paramètre spécial **$?**. 
-  * L'​opérateur **<​nowiki>&&</​nowiki>​** exécute la commande suivante, si la commande précédente à renvoyée le code retour true (égale à 0). 
-  * L'​opérateur **<​nowiki>​||</​nowiki>​** exécute la commande suivante, si la commande précédente à renvoyée le code retour false (différent de 0). 
  
 +  *  Une commande de test renvoie le code de retour 0 (considérer comme vrai) lorsque le test réussi et 1 (considérer comme faux) lorsqu'​il échoue.
 +  *  Le code retour d'une commande est mémorisé dans le paramètre spécial $?.
 +  *  L'​opérateur de contrôle **&&​** exécute la commande suivante, si la commande précédente à renvoyée un code de retour égale à 0.
 +  *  L'​opérateur de contrôle **||** exécute la commande suivante, si la commande précédente à renvoyée un code de retour supérieur à 0.
 +</​note>​
 +
 +===== Comparaison numérique avec [ et [[ =====
 +
 +Les commandes de test dispose de 6 primitives binaires pour effectuer des comparaisons numériques.
 +
 +==== Syntaxe ====
 +
 +  * **test expr1 OP expr2**
 +  * **[ expr1 OP expr2 ]**
 +  * **<​nowiki>​[[</​nowiki>​ expr1 OP <​nowiki>​expr2</​nowiki>​ ]]**
 +
 +  * Avec :
 +    * **Expr1** et **expr2** sont sujets au développement des paramètres.
 +    * Avec la commande **<​nowiki>​[[</​nowiki>​**,​ **expr1** et **expr2** sont sujets à l’évaluation arithmétique. Les commandes **[** ou **test**, ne le permettent pas.
 +    * **OP**, l'une des primitives du tableau suivant.
 +
 +|  Liste des primitives de comparaison numérique ​ ||
 +^Primitives ^ Retours ^
 +|''​-eq ''​ | est égal à |
 +|''​-ne ''​ | n'est pas égal à |
 +|''​-gt ''​ | est plus grand que |
 +|''​-ge ''​ |est plus grand ou égal à |
 +|''​-lt ''​ |est plus petit que |
 +|''​-le ''​ |est plus petit ou égal à |
 +
 +==== Exemples ====
 +
 +<code user>
 +[ 25 -eq 20 ] # Teste si 25 est égale à 20
 +echo $?
 +</​code><​code>​
 +
 +</​code>​
 +
 +<code user>
 +var1=17
 +var2=18
 +echo "$var1 est-il plus grand que $var2 : $([ $var1 -gt $var2 ]; echo $?)"
 +echo "$var1 est-il plus petit que $var2 : $([[ $var1 -lt $var2 ]]; echo $?)"
 +echo "$var1 est-il différent de $var2 : $(test $var1 -ne $var2 ; echo $?)"
 +
 +unset var1 var2
 +</​code><​code>​
 +17 est-il plus grand que 18 : 1
 +17 est-il plus petit que 18 : 0
 +17 est-il différent de 18 : 0
 +</​code>​
 +
 +Copions le code ci-dessous dans le fichier **mon_script**.\\ ​
 +Puis exécutons le avant de le supprimer.
 +
 +<code bash mon_script>​
 +#!/bin/bash
 +a=2 ; b=1
 +if [ "​$a"​ -gt "​$b"​ ] ; then
 + echo "$a est plus grand que $b"
 +fi
 +
 +if test 100 -gt 99
 +then
 + echo "​vrai" ​  #​réponse : vrai
 +else
 + echo "​faux"​
 +fi    ​
 +</​code><​code user>
 +bash mon_script
 +
 +rm -f mon_script
 +</​code><​file>​
 +2 est plus grand que 1
 +vrai
 +'​mon_script'​ supprimé
 +</​file>​
 +
 +==== Particularités de la commande [[ ====
 +
 +Seule la commande **<​nowiki>​[[</​nowiki>​** permet de tester une expression arithmétique.
 +
 +<code user>
 +test 21+21 -eq 42 ; echo -e $?\\n
 +
 +[ 21*2 -eq 42 ] ; echo -e $?\\n
 +
 +[[ 84-42 -eq 42 ]] ; echo $?
 +</​code><​code>​
 +bash: test: 21+21 : nombre entier attendu comme expression
 +2
 +
 +bash: [: 21*2 : nombre entier attendu comme expression
 +2
 +
 +0
 +</​code>​
 +<​note>​
 +Lorsqu'​une commande interne **le code de retour renvoie 2**, cela signifie un mauvais usage de cette commande.
 +</​note>​
 +
 +Elle est aussi la seule à accepter les chaînes vides qui sont alors évaluées à **0**.
 +
 +<code user>
 +[[ 0 -eq ""​ ]] ; echo $?
 +[[ ""​ -ne 0 ]] ; echo $?
 +[[ ""​ -eq ""​ ]] ; echo $?
 +</​code><​code>​
 +0
 +1
 +0
 +</​code>​
 +
 +Idem pour les chaînes de caractères sans espace (mots). Un mot est substitué par **0**.
 +
 +<code user>
 +[[ mot -eq 0 ]] ; echo $?
 +[[ 0 -ne mot ]] ; echo $?
 +[[ mot1 -eq mot2 ]] ; echo $?
 +</​code><​code>​
 +0
 +1
 +0
 +</​code>​
 +
 +<note important>​
 +Attention :
 +<code bash mon_script>​
 +#!/bin/bash
 +var1=8
 +var2=7
 +if test "​$e"​ -gt "​$f"​ ; then
 +  echo " $var1 est plus grand que $var2 "
 +fi
 +printf \\n
 +test 8 -gt 7 && echo "$var1 est plus grand que $var2"
 +</​code>​
 +
 +<code user>
 +bash mon_script
 +
 +rm -v mon_script
 +</​code>​
 +<file config retour de la commande>​
 +mon_script: ligne 4 : test:  : nombre entier attendu comme expression
 +
 +8 est plus grand que 7 
 +'​mon_script'​ supprimé
 +</​file>​
 +</​note>​
 +
 +==== Mauvais usages ====
 +
 +Le caractère **$** est obligatoire pour développer les paramètres. Sans, ils sont interprétés comme des mots.
 +
 +Avec la commande **[** (ou **test**), les chaînes vides ou les mots sont à proscrire.
 +
 +<code user>
 +test 42 -eq ""​ ; echo $?
 +[ 42 -lt ""​ ] ; echo $?
 +echo =====
 +test ""​ -gt 42 ; echo $?
 +[ ""​ -ge 42 ] ; echo $?
 +</​code><​code>​
 +bash: test:  : nombre entier attendu comme expression
 +2
 +bash: [:  : nombre entier attendu comme expression
 +2
 +=====
 +bash: test:  : nombre entier attendu comme expression
 +2
 +bash: [:  : nombre entier attendu comme expression
 +2
 +</​code>​
 +
 +<code user>
 +test mot -eq 0 ; echo $?
 +[ 0 -eq mot ] ; echo $?
 +test O -le 0 ; echo $?
 +[ "​mot"​ -ge 0 ] ; echo $?
 +echo $?
 +</​code><​code>​
 +bash: test: mot : nombre entier attendu comme expression
 +2
 +bash: [: mot : nombre entier attendu comme expression
 +2
 +bash: test: O : nombre entier attendu comme expression
 +2
 +bash: [: mot : nombre entier attendu comme expression
 +2
 +</​code>​
 +
 +Avec **[** (ou **test**) et **<​nowiki>​[[</​nowiki>​**,​ les opérandes ne peuvent être des chaînes de caractères qui comportent des espaces.
 +
 +<code user>
 +test "Du texte" -eq 0 ; echo $?
 +[ 0 -ge "Du texte" ] ; echo $?
 +[[ "Du texte" -ge 0 ]] ; echo $?
 +</​code><​code>​
 +bash: test: Du texte : nombre entier attendu comme expression
 +2
 +bash: [: Du texte : nombre entier attendu comme expression
 +2
 +bash: [[: Du texte : erreur de syntaxe dans l'​expression (le symbole erroné est « texte »)
 +1
 +</​code>​
 +
 +<note >
 +Il est remarquable que le test ''<​nowiki>​[[ "Du texte" -ge 0 ]]</​nowiki>'',​ ne renvoie pas le **code de retour 2**.\\ ​
 +Pourtant il s'agit bien d'un mauvais usage de la commande ''<​nowiki>​[[</​nowiki>''​.
 +</​note>​
 +
 +Les deux opérandes sont obligatoires.
 +
 +<code user>
 +test 42 -eq ; echo $?
 +[ 42 -lt ] ; echo $?
 +[[ 42 -ne ]]
 +echo $?
 +echo =====
 +test -gt 42 ; echo $?
 +[ -ge 42 ] ; echo $?
 +[[ -le 42 ]]
 +echo $?
 +</​code><​code>​
 +bash: test: 42 : opérateur unaire attendu
 +2
 +bash: [: 42 : opérateur unaire attendu
 +2
 +bash: argument « ]] » inattendu pour l'​opérateur binaire conditionnel
 +bash: erreur de syntaxe près de « ]] »
 +2
 +=====
 +bash: test: -gt : opérateur unaire attendu
 +2
 +bash: [: -ge : opérateur unaire attendu
 +2
 +bash: opérateur binaire conditionnel attendu
 +bash: erreur de syntaxe près de « 42 »
 +2
 +</​code>​
 +
 +Ainsi que les espaces.\\ ​
 +Lorsqu'​il n'y pas d'​espaces entre les opérandes et l'​opérateur,​ l'​ensemble est considéré comme une chaîne de caractères\\ ​
 +Puisque la chaîne est non vide, le test n'​échoue pas.
 +
 +<code user>
 +test 420-ne420 ; echo $?
 +[ 420-gt42 ] ; echo $?
 +[[ 42-lt420 ]] ; echo $?
 +</​code><​code>​
 +0
 +0
 +0
 +</​code>​
 +
 +Ne pas utiliser les opérateurs **<** et **>** avec **[** (ou **test**) et **<​nowiki>​[[</​nowiki>​** pour réaliser des comparaisons numériques.\\ ​
 +Car avec ces commandes, ces opérateur sont des opérateurs de comparaison lexicographique.\\ ​
 +(voir [[https://​debian-facile.org/​doc:​programmation:​shells:​la-page-man-bash-ii-les-operateurs-lexicographiques|Bash : Les opérateurs sur chaînes]]).
 +
 +__Rappel :__ Avec **[** (ou **test**), les opérateurs **<** et **>** s'​utilisent protégés (voir [[https://​debian-facile.org/​atelier:​chantier:​bash-comparaison-lexicographique-des-chaines#​tests-avec1|Bash,​ comparaison lexicographique avec [ ou test]]).
 +<code user>
 +test 425 \> 4242 ; echo $?
 +[ 426 \> 4242 ] ; echo $?
 +[[ 4242 < 427 ]] ; echo $?
 +</​code>​
 +<file config retour des commandes>​
 +0
 +0
 +0
 +</​file>​
 +
 +Lexicographiquement 425, 426 et 427 sont supérieurs (placés après dans l'​ordre lexicographique),​ mais sont inférieurs (plus petits) numériquement à 4242.
  
 ===== Comparaison numérique avec (( ===== ===== Comparaison numérique avec (( =====
Ligne 43: Ligne 327:
 <​code>​ <​code>​
 let <​expr1><​OP><​expr2>​ let <​expr1><​OP><​expr2>​
 +let "<​expr1>​ <OP> <​expr2>"​
 (( <​expr1>​ <OP> <​expr2>​ )) (( <​expr1>​ <OP> <​expr2>​ ))
  
 <​expr1>​ et <​expr2>​ sont sujets au développement des paramètres et à l’évaluation arithmétique. <​expr1>​ et <​expr2>​ sont sujets au développement des paramètres et à l’évaluation arithmétique.
 Avec <OP> l'un des opérateurs donnés dans le tableau suivant. Avec <OP> l'un des opérateurs donnés dans le tableau suivant.
-Avec la commande **let**, les opérateurs commençant par **<** ou **>** doivent être protégés (voir plus bas : Mauvais usages). 
 </​code>​ </​code>​
  
-__Remarque ​:__ Les commandes **let** et **<​nowiki>​((</​nowiki>​** sont pratiquement ​équivalentes ​(voir les différences plus bas Mauvais usages).+__Remarques ​:__ Les commandes **let** et **<​nowiki>​((</​nowiki>​** sont équivalentes.
  
-|  Opérateurs de comparaison numérique ​de la commande ​<​nowiki>​((</​nowiki> ​ ||+|  Opérateurs de comparaison numérique ​des commandes let et <​nowiki>​((</​nowiki> ​ ||
 ^ Opérateurs ^ Description ^ ^ Opérateurs ^ Description ^
 | ''​ == ''​ | Comparaison d'une égalité ​             | | ''​ == ''​ | Comparaison d'une égalité ​             |
Ligne 60: Ligne 344:
 | ''​ < ''​ | Comparaison de plus petit que         | | ''​ < ''​ | Comparaison de plus petit que         |
 | ''<​nowiki><​=</​nowiki>​ ''​ | Comparaison de plus petit ou égal que | | ''<​nowiki><​=</​nowiki>​ ''​ | Comparaison de plus petit ou égal que |
 +
 +
 +La commande **let** ne supporte qu'un seul argument.\\ ​
 +Si l'​expression arithmétique à évaluer comporte des espaces, l'​expression doit être protégée par des guillemets simples ou doubles.\\ ​
 +Si l'​expression n'est pas protégée, les opérateurs commençant par **<** ou **>** doivent être protégés\\ ​
 +(Voir plus bas : [[https://​debian-facile.org/​doc:​programmation:​shells:​page-man-bash-iii-les-operateurs-de-comparaison-numerique#​mauvais-usages|Mauvais usages]].)
  
 ==== Exemples dans le terminal ==== ==== Exemples dans le terminal ====
Ligne 79: Ligne 369:
 (( $nombre1 > $nombre2 )) # Est-ce-que nombre1 (12) est strictement supérieur au nombre2 (13). (( $nombre1 > $nombre2 )) # Est-ce-que nombre1 (12) est strictement supérieur au nombre2 (13).
 echo $? # Affiche le code de retour. echo $? # Affiche le code de retour.
-let $nombre1!=$nombre2 #Est-ce-que nombre1 (12) est différent du nombre2 (13)+let "$nombre1 != $nombre2" #Est-ce-que nombre1 (12) est différent du nombre2 (13)
 echo $? # Affiche le code de retour. echo $? # Affiche le code de retour.
  
Ligne 106: Ligne 396:
 Comme nous l'​avons vu au dessus, les paramètres n'ont pas besoin du caractère **$** pour être développés.\\ ​ Comme nous l'​avons vu au dessus, les paramètres n'ont pas besoin du caractère **$** pour être développés.\\ ​
 Un simple mot sera alors interprété comme un paramètre.\\ ​ Un simple mot sera alors interprété comme un paramètre.\\ ​
-Si ce paramètre n'​existe pas, son développement renverra ​**0**.+Si ce paramètre n'​existe pas, le mot sera substitué par **0**.
  
 <code user> <code user>
 (("​mot"​ == 0)) (("​mot"​ == 0))
 echo '​(("​mot"​ == 0)) revoie le code de retour'​ $? echo '​(("​mot"​ == 0)) revoie le code de retour'​ $?
-let mot\<=0 +let "mot <= 0" 
-echo 'let "mot"\<=0 revoie le code de retour'​ $?+echo 'let "mot <= 0" ​revoie le code de retour'​ $?
  
 n="​mot"​ n="​mot"​
Ligne 124: Ligne 414:
 <file config retour des commandes>​ <file config retour des commandes>​
 (("​mot"​ == 0)) revoie le code de retour 0 (("​mot"​ == 0)) revoie le code de retour 0
-let "mot"\<=0 revoie le code de retour 0+let "mot <= 0" ​revoie le code de retour 0
 Avec n="​mot"​ ; ((n!=0)) renvoie le code de retour 1 Avec n="​mot"​ ; ((n!=0)) renvoie le code de retour 1
 Avec n="​mot"​ ; let n==0 renvoie le code de retour 0 Avec n="​mot"​ ; let n==0 renvoie le code de retour 0
 </​file>​ </​file>​
  
 +Si un paramètre existe mais que sa valeur est vide, son développement retournera 0.
 +<code user>
 +n=
 +((n==0)) ; echo $?
 +let 'n != 0' ; echo $?
 +
 +unset n
 +</​code>​
 +<file config retour des commandes>​
 +0
 +1
 +</​file>​
 Avec la commande **<​nowiki>​((</​nowiki>​**,​ les espaces ne sont pas obligatoires. Avec la commande **<​nowiki>​((</​nowiki>​**,​ les espaces ne sont pas obligatoires.
 <code user> <code user>
Ligne 148: Ligne 450:
 ==== Mauvais usages ==== ==== Mauvais usages ====
  
-La commande **let** ​ne supporte ​pas les espaces entre les opérandes et l'​opérateur.+Avec la commande **let**, si l'​expression à évaluer n'​est ​pas protégée,​\\  
 +Les espaces entre les opérandes et l'​opérateur ​ne sont pas supportés.
  
 <code user> <code user>
Ligne 161: Ligne 464:
 </​file>​ </​file>​
  
-Avec la commande **let**, les opérateurs **<**, **<​nowiki><​=</​nowiki>​**,​ **>** et **>=** doivent être protégés.\\ 
  
 +Avec la commande **let**, si l'​expression à évaluer est protégé par des guillemets simples,​\\ ​
 +Les paramètres sont développés uniquement s'ils ne disposent pas du caractère **$**.\\ ​
 +S'il est fourni, une erreur est renvoyée.
 +
 +<code user>
 +n=42
 +let '​n==42'​ ;​ echo $?
 +let '$n > 42' ; echo $?
 +</​code>​
 +<file config retour des commandes>​
 +0
 +bash: let: $n == 42 : erreur de syntaxe : opérande attendu (le symbole erroné est « $n == 42 »)
 +1
 +</​file>​
 +
 +Avec la commande **let**, si l'​expression à évaluer n'est pas protégée, les opérateurs **<**, **<​nowiki><​=</​nowiki>​**,​ **>** et **>=** doivent être protégés.\\
 Les trois types de protection (**\**, entre guillemets simples **<​nowiki>''</​nowiki>​** et entre guillemets doubles **""​** fonctionnent.\\ ​ Les trois types de protection (**\**, entre guillemets simples **<​nowiki>''</​nowiki>​** et entre guillemets doubles **""​** fonctionnent.\\ ​
 S'ils ne sont pas protégés, les opérateurs **<** et **>** sont des opérateurs de redirection.\\ ​ S'ils ne sont pas protégés, les opérateurs **<** et **>** sont des opérateurs de redirection.\\ ​
Ligne 236: Ligne 554:
 Sans guillemets, il est nécessaire de protéger le caractère de protection (**\\n**). Sans guillemets, il est nécessaire de protéger le caractère de protection (**\\n**).
  
-<​code>​ +<​code ​user>
-echo =====+
 (("Du texte" >= 0)) (("Du texte" >= 0))
-echo -e '​(("​Du texte" >= 0)) renvoie le code de retour'​ $?\\n+echo -e '(("Du texte" >= 0)) renvoie le code de retour'​ $?\\n
 let 424\>"​Du texte" let 424\>"​Du texte"
-echo -e 'let 424\>"​Du texte" renvoie le code de retour'​ $?\\n +echo -e 'let 424\>"​Du texte" renvoie le code de retour'​ $?\\n 
-echo =====+
 n="Du texte" n="Du texte"
 ((0<=n)) ((0<=n))
-echo -e "Avec n=\"​$n\"​ ; ((0<=n)) renvoie le code retour $?\n"+echo -e "Avec n=\"​$n\"​ ; ((0<=n)) renvoie le code retour $?\n"
 let n==0 let n==0
-echo "Avec n=\"​${n}\"​ ; let n==0 renvoie le code retour $?"+echo "Avec n=\"​${n}\"​ ; let n==0 renvoie le code retour $?" 
 + 
 +unset n
 </​code>​ </​code>​
-<file config retour des commandes>​=====+<file config retour des commandes>​
 bash: ((: Du texte >= 0 : erreur de syntaxe dans l'​expression (le symbole erroné est « texte >= 0 ») bash: ((: Du texte >= 0 : erreur de syntaxe dans l'​expression (le symbole erroné est « texte >= 0 »)
-(("Du texte" >= 0)) renvoie le code de retour 1+(("Du texte" >= 0)) renvoie le code de retour 1
  
 bash: let: 424>Du texte : erreur de syntaxe dans l'​expression (le symbole erroné est « texte ») bash: let: 424>Du texte : erreur de syntaxe dans l'​expression (le symbole erroné est « texte »)
-let 424\>"​Du texte" renvoie le code de retour 1+let 424\>"​Du texte" renvoie le code de retour 1
  
-===== 
 bash: ((: Du texte : erreur de syntaxe dans l'​expression (le symbole erroné est « texte ») bash: ((: Du texte : erreur de syntaxe dans l'​expression (le symbole erroné est « texte »)
-Avec n="Du texte" ; ((0<=n)) renvoie le code retour 1+Avec n="Du texte" ; ((0<=n)) renvoie le code retour 1
  
 bash: let: Du texte : erreur de syntaxe dans l'​expression (le symbole erroné est « texte ») bash: let: Du texte : erreur de syntaxe dans l'​expression (le symbole erroné est « texte »)
--Avec n="Du texte" ; let n==0 renvoie le code retour 1+Avec n="Du texte" ; let n==0 renvoie le code retour 1
 </​file>​ </​file>​
  
Ligne 346: Ligne 664:
 0 0
 </​file>​ </​file>​
-===== Comparaison numérique avec [ et [[ ===== 
-La comparaison numérique peut aussi se réaliser avec les commandes **[** (ou **test**) et **<​nowiki>​[[</​nowiki>​**. 
- 
-Syntaxe : 
-<​code>​ 
-test expr1 OP expr2 
-[ expr1 OP expr2 ] 
-[[ expr1 OP expr2 ]] 
-  ​ 
-expr1 et expr2 sont sujets au développement des paramètres. 
-Avec la commande [[, expr1 et expr2 sont sujets à l’évaluation arithmétique. La commande [ (ou test), ne le permet pas. 
-Avec OP, l'un des opérateurs du tableau suivant. 
-</​code>​ 
- 
-__Rappel :__ Les commandes **[** et **test** sont équivalentes. 
- 
-|  Opérateurs de comparaison des commandes [ et <​nowiki>​[[</​nowiki> ​ || 
-^Opérateurs ​  ​^ ​ Significations ​        ^ 
-|''​-eq ''​ | est égal à | 
-|''​-ne ''​ | n'est pas égal à | 
-|''​-gt ''​ | est plus grand que | 
-|''​-ge ''​ |est plus grand ou égal à | 
-|''​-lt ''​ |est plus petit que | 
-|''​-le ''​ |est plus petit ou égal à | 
- 
-==== Exemples dans le terminal ==== 
- 
-<code user> 
-[ 25 -eq 20 ] # Teste si 25 est égale à 20 
-echo $? 
-</​code>​ 
-<file config retour de la commande>​ 
- 
-</​file>​ 
- 
-<code user> 
-var1=17 
-var2=18 
-echo "$var1 est-il plus grand que $var2 : $([ $var1 -gt $var2 ]; echo $?)" 
-echo "$var1 est-il plus petit que $var2 : $([[ $var1 -lt $var2 ]]; echo $?)" 
-echo "$var1 est-il différent de $var2 : $(test $var1 -ne $var2 ; echo $?)" 
- 
-unset var1 var2 
-</​code>​ 
-<file config retour de la commande>​ 
-17 est-il plus grand que 18 : 1 
-17 est-il plus petit que 18 : 0 
-17 est-il différent de 18 : 0 
-</​file>​ 
- 
-==== Particularités de la commande [[ ==== 
- 
-Seule la commande **<​nowiki>​[[</​nowiki>​** permet de tester une expression arithmétique. 
- 
-<code user> 
-test 21+21 -eq 42 ; echo -e $?\\n 
- 
-[ 21*2 -eq 42 ] ; echo -e $?\\n 
- 
-[[ 84-42 -eq 42 ]] ; echo $? 
-</​code>​ 
-<file config retour des commandes>​ 
-bash: test: 21+21 : nombre entier attendu comme expression 
-2 
- 
-bash: [: 21*2 : nombre entier attendu comme expression 
-2 
- 
-0 
-</​file>​ 
- 
-__Rappel :__ Lorsque le code de retour d'une commande interne renvoie **2**, cela signifie un mauvais usage de cette commande. 
- 
-Elle est aussi la seule à accepter les chaînes vides qui sont alors évaluées à **0**. 
- 
-<code user> 
-[[ 0 -eq ""​ ]] ; echo $? 
-[[ ""​ -ne 0 ]] ; echo $? 
-[[ ""​ -eq ""​ ]] ; echo $? 
-</​code>​ 
-<file config retour des commande>​ 
-0 
-1 
-0 
-</​file>​ 
- 
-Idem pour les chaînes de caractères sans espace (mots). Un mot est substitué par **0**. 
- 
-<code user> 
-[[ mot -eq 0 ]] ; echo $? 
-[[ 0 -ne mot ]] ; echo $? 
-[[ mot1 -eq mot2 ]] ; echo $? 
-</​code>​ 
-<file config retour des commandes>​ 
-0 
-1 
-0 
-</​file>​ 
- 
-==== Mauvais usages ==== 
- 
-Le caractère **$** est obligatoire pour développer les arguments. Sans, ils sont interprétés comme des mots. 
- 
-Avec la commande **[** (ou **test**), les chaînes vides ou les mots sont à proscrire. 
- 
-<code user> 
-test 42 -eq ""​ ; echo $? 
-[ 42 -lt ""​ ] ; echo $? 
-echo ===== 
-test ""​ -gt 42 ; echo $? 
-[ ""​ -ge 42 ] ; echo $? 
-</​code>​ 
-<file config retour des commande>​ 
-bash: test:  : nombre entier attendu comme expression 
-2 
-bash: [:  : nombre entier attendu comme expression 
-2 
-===== 
-bash: test:  : nombre entier attendu comme expression 
-2 
-bash: [:  : nombre entier attendu comme expression 
-2 
-</​file>​ 
- 
-<code user> 
-test mot -eq 0 ; echo $? 
-[ 0 -eq mot ] ; echo $? 
-test O -le 0 ; echo $? 
-[ "​mot"​ -ge 0 ] ; echo $? 
-echo $? 
-</​code>​ 
-<file config retour des commandes>​ 
-bash: test: mot : nombre entier attendu comme expression 
-2 
-bash: [: mot : nombre entier attendu comme expression 
-2 
-bash: test: O : nombre entier attendu comme expression 
-2 
-bash: [: mot : nombre entier attendu comme expression 
-2 
-</​file>​ 
- 
-Avec **[** (ou **test**) et **<​nowiki>​[[</​nowiki>​**,​ les opérandes ne peuvent être des chaînes de caractères qui comportent des espaces. 
- 
-<code user> 
-test "Du texte" -eq 0 ; echo $? 
-[ 0 -ge "Du texte" ] ; echo $? 
-[[ "Du texte" -ge 0 ]] ; echo $? 
-</​code>​ 
-<file config retour des commandes>​ 
-bash: test: Du texte : nombre entier attendu comme expression 
-2 
-bash: [: Du texte : nombre entier attendu comme expression 
-2 
-bash: [[: Du texte : erreur de syntaxe dans l'​expression (le symbole erroné est « texte ») 
-1 
-</​file>​ 
-<note > 
-Il est remarquable que le test ''<​nowiki>​[[ "Du texte" -ge 0 ]]</​nowiki>'',​ ne renvoie pas le code de retour **2**.\\ ​ 
-Pourtant il s'agit bien d'un mauvais usage de la commande ''<​nowiki>​[[</​nowiki>''​. 
-</​note>​ 
- 
-Les deux opérandes sont obligatoires. 
- 
-<code user> 
-test 42 -eq ; echo $? 
-[ 42 -lt ] ; echo $? 
-[[ 42 -ne ]] 
-echo $? 
-echo ===== 
-test -gt 42 ; echo $? 
-[ -ge 42 ] ; echo $? 
-[[ -le 42 ]] 
-echo $? 
-</​code>​ 
-<file config retour des commandes>​ 
-bash: test: 42 : opérateur unaire attendu 
-2 
-bash: [: 42 : opérateur unaire attendu 
-2 
-bash: argument « ]] » inattendu pour l'​opérateur binaire conditionnel 
-bash: erreur de syntaxe près de « ]] » 
-2 
-===== 
-bash: test: -gt : opérateur unaire attendu 
-2 
-bash: [: -ge : opérateur unaire attendu 
-2 
-bash: opérateur binaire conditionnel attendu 
-bash: erreur de syntaxe près de « 42 » 
-2 
-</​file>​ 
- 
-Ainsi que les espaces. 
- 
-<code user> 
-[24 -ne 24 ] ; echo $? 
-[[24 -le 24 ]] 
-echo $? 
-echo ==== 
-[ 24 -ne 24] ; echo $? 
-[[ 24 -le 24]] 
-echo $? 
-echo ==== 
-test 42-gt 0 ; echo $? 
-[ 42 -eq0 ] ; echo $? 
-[[ 42 -ne0 ]] 
-echo $? 
-</​code>​ 
-<file config retour des commandes>​ 
-bash: [24 : commande introuvable 
-127 
-bash: [[24 : commande introuvable 
-127 
-==== 
-bash: [: « ] » manquant 
-2 
-bash: erreur de syntaxe dans une expression conditionnelle 
-bash: erreur de syntaxe près de « echo » 
-==== 
-bash: test: 42-gt : opérateur unaire attendu 
-2 
-bash: [: 42 : opérateur unaire attendu 
-2 
-bash: opérateur binaire conditionnel attendu 
-bash: erreur de syntaxe près de « -ne0 » 
-2 
-</​file>​ 
- 
-__Rappel :__ Le code retour 127, signifie que la commande n'a pas été trouvée. 
- 
-Lorsqu'​il n'y pas d'​espaces entre les opérandes et l'​opérateur,​ l'​ensemble est considéré comme une chaîne de caractères\\ ​ 
-Puisque la chaîne est non vide, le test n'​échoue pas. 
-<code user> 
-test 420-ne420 ; echo $? 
-[ 420-gt42 ] ; echo $? 
-[[ 42-lt420 ]] ; echo $? 
-</​code>​ 
-<file config retour des commandes>​ 
-0 
-0 
-0 
-</​file>​ 
- 
-Ne pas utiliser les opérateurs **<** et **>** avec **[** (ou **test**) et **<​nowiki>​[[</​nowiki>​** pour réaliser des comparaisons numériques.\\ ​ 
-Car avec ces commandes, ces opérateur sont des opérateurs de comparaison lexicographique.\\ ​ 
-(voir [[https://​debian-facile.org/​doc:​programmation:​shells:​la-page-man-bash-ii-les-operateurs-lexicographiques|Bash : les opérateurs lexicographiques]]). 
- 
-__Rappel :__ Avec **[** (ou **test**), les opérateurs **<** et **>** s'​utilisent protégés (voir [[https://​debian-facile.org/​atelier:​chantier:​bash-comparaison-lexicographique-des-chaines#​tests-avec1|Bash,​ comparaison lexicographique avec [ ou test]]). 
-<code user> 
-test 425 \> 4242 ; echo $? 
-[ 426 \> 4242 ] ; echo $? 
-[[ 4242 < 427 ]] ; echo $? 
-</​code>​ 
-<file config retour des commandes>​ 
-0 
-0 
-0 
-</​file>​ 
- 
-Lexicographiquement 425, 426 et 427 sont supérieurs (placés après dans l'​ordre lexicographique),​ mais sont inférieurs (plus petits) numériquement à 4242. 
- 
-==== Exemples dans un script ==== 
- 
-Copions le code ci-dessous dans le fichier mon_script. 
- 
-<code bash mon_script>​ 
-#!/bin/bash 
-a=2 ; b=1 
-if [ "​$a"​ -gt "​$b"​ ] ; then 
- echo "$a est plus grand que $b" 
-fi 
- 
-if test 100 -gt 99 
-then 
- echo "​vrai" ​  #​réponse : vrai 
-else 
- echo "​faux"​ 
-fi    ​ 
-</​code>​ 
- 
-<code user> 
-bash mon_script 
- 
-rm -f mon_script 
-</​code>​ 
-<file config retour de la commande>​ 
-2 est plus grand que 1 
-vrai 
-'​mon_script'​ supprimé 
-</​file>​ 
- 
-<note important>​ 
-Attention : 
-<code bash mon_script>​ 
-#!/bin/bash 
-var1=8 
-var2=7 
-if test "​$e"​ -gt "​$f"​ ; then 
-  echo " $var1 est plus grand que $var2 " 
-fi 
-printf \\n 
-test 8 -gt 7 && echo "$var1 est plus grand que $var2" 
-</​code>​ 
- 
-<code user> 
-bash mon_script 
- 
-rm -v mon_script 
-</​code>​ 
-<file config retour de la commande>​ 
-mon_script: ligne 4 : test:  : nombre entier attendu comme expression 
- 
-8 est plus grand que 7  
-'​mon_script'​ supprimé 
-</​file>​ 
-</​note>​ 
-=====Conclusion sur les opérateurs lexicographiques et les opérateurs de comparaison numérique===== 
-====Yep, plus de confusions ! ==== 
-<​note>​ 
-Nous avons vu :\\  
- 
-  * qu'il y a deux mécanismes pour effectuer des tests, la paire de crochets et la commande test;  
- 
-  * qu'à chaque mécanisme sa syntaxe :  
-<​code>​ 
-$ test -f /etc/gaga 
-</​code>​ 
- 
-Ou  
- 
-<​code>​ 
-$ [ -f /etc/gaga ] 
-</​code>​ 
-(Si le fichier /etc/gaga existait, //le code de retour// serait 0.\\ Voir la commande exit et son utilisation dans les scripts.) 
- 
-  * qu'il était possible de faire des tests avec commande test ou les crochets sur [[doc:​programmation:​shells:​la-page-man-bash-ii-les-operateurs-lexicographiques#​caracteres-de-comparaison-sur-les-chaines-de-caracteres|les chaînes de caractères]] ; et [[doc:​programmation:​shells:​la-page-man-bash-ii-les-operateurs-lexicographiques#​tests-sur-les-fichiers|les fichiers]]. ​ 
- 
-  * que, pour des comparaisons sur [[doc:​programmation:​shells:​page-man-bash-iii-les-operateurs-de-comparaison-numerique|les nombres]] on utilise : 
-soit les doubles parenthèses avec les opérateurs booléens : 
-<​code>​ 
- ​== ​   !=   <​ > ​  >​= ​  <= 
-</​code>​ 
-soit les crochets (ou la commande "​test"​) avec les opérateurs :  
-<​code>​ 
--eq   ​-ne ​  ​-gt ​  ​-ge ​  ​-lt ​  -le 
-</​code>​ 
- 
-  * qu'on utilise exclusivement les opérateurs ''​-eq''​ ; ''​-ne''​ ; ''​-gt''​ ; ''​-ge''​ ; ''​-lt''​ ; ''​-le''​ sur les nombres mais pas sur les chaînes de caractères ; 
- 
-  * qu'​enfin il ne faut pas confondre pas avec les opérateurs précédents,​ les options qu'on utilise exclusivement pour tester les fichiers :''​-e'' ​ ; ''​-d'';​ ''​-f''​ ; ''​-L''​ ; ''​-r''​ ; ''​-w''​ ; ''​-x''​ ; $fichier1 ''​-nt''​ $fichier2 ; $fichier1 ''​-ot''​ $fichier2 
- 
-Plein d'​exemples dans ce wiki en anglais : [[http://​wiki.bash-hackers.org/​commands/​classictest]] 
-</​note>​ 
  
-:-D 
  
 =====Pour aller plus loin : les opérateurs logiques.===== =====Pour aller plus loin : les opérateurs logiques.=====
doc/programmation/shells/page-man-bash-iii-les-operateurs-de-comparaison-numerique.txt · Dernière modification: 30/09/2023 23:06 (modification externe)

Pied de page des forums

Propulsé par FluxBB