7.3. Autres opérateurs de comparaison

Un opérateur de comparaison binaire compare deux variables ou quantités. Notez que la comparaison entre entiers et chaînes utilisent un différent ensemble d'opérateurs.

comparaison d'entiers

-eq

est égal à

if [ "$a" -eq "$b" ]

-ne

n'est pas égal à

if [ "$a" -ne "$b" ]

-gt

est plus grand que

if ["$a" -gt "$b" ]

-ge

est plus grand ou égal à

if [ "$a" -ge "$b" ]

-lt

est plus petit que

if [ "$a" -lt "$b" ]

-le

est plus petit ou égal à

if [ "$a" -le "$b" ]

<

est plus petit que (à l'intérieur de parenthèses doubles)

(("$a" < "$b"))

<=

est plus petit ou égal à (à l'intérieur de parenthèses doubles)

(("$a" <= "$b"))

>

est plus grand que (à l'intérieur de parenthèses doubles)

(("$a" > "$b"))

>=

est plus grand ou égal à (à l'intérieur de parenthèses doubles)

(("$a" >= "$b"))

comparaison de chaînes de caractères

=

est égal à

if [ "$a" = "$b" ]

==

est égal à

if [ "$a" == "$b" ]

Ceci est un synonyme de =.

[Note]

Note

L'opérateur de comparaison == se comporte différemment à l'intérieur d'un test à double crochets qu'à l'intérieur de crochets simples.

[[ $a == z* ]]    # Vrai si $a commence avec un "z" (correspondance de modèle).

[[ $a == "z*" ]]  # Vrai si $a est égal à z* (correspondance littérale).

[ $a == z* ]      # Correspondance de fichiers et découpage de mots.
[ "$a" == "z*" ]  # Vrai si $a est égal à z* (correspondance littérale).

# Merci, Stéphane Chazelas
!=

n'est pas égal à

if [ "$a" != "$b" ]

Cet opérateur utilise la reconnaissance de motifs à l'intérieur de constructions [[ ... ]].

<

est plus petit que, d'après l'ordre alphabétique ASCII

if [[ "$a" &lt; "$b" ]]

if [ "$a" \< "$b" ]

Notez que « < » a besoin d'être dans une séquence d'échappement s'il se trouve à l'intérieur de [ ].

>

est plus grand que, d'après l'ordre alphabétique ASCII

if [[ "$a" > "$b" ]]

if [ "$a" \> "$b" ]

Notez que « > » a besoin d'être dans une séquence d'échappement s'il se trouve à l'intérieur de [ ].

Voir l'Exemple 26.10, « Le tri bulle : Bubble Sort » pour une application de cet opérateur de comparaison.

-n

la chaîne de caractères n'est pas « vide ».

[Attention]

Attention

Attention : Le test -n nécessite absolument que la chaîne de caractères soit entre guillemets à l'intérieur des crochets de test. Utiliser une chaîne sans guillemets avec ! -z, voire simplement la chaîne sans guillemets à l'intérieur des crochets (voir l'Exemple 7.6, « Vérification si une chaîne est nulle ») fonctionne habituellement, néanmoins, c'est une pratique peu sûre. Placez toujours vos chaînes de caractères à tester entre guillemets. [26]

-z

la chaîne de caractères est « vide », c'est-à-dire qu'elle a une taille nulle

Exemple 7.5. Comparaisons de nombres et de chaînes de caractères

#!/bin/bash

a=4
b=5

#  Ici, "a" et "b" peuvent être traités soit comme des entiers soit comme des
#+ chaînes de caractères.
#  Il y a un peu de flou entre les comparaisons arithmétiques et de chaînes de
#+ caractères car les variables Bash ne sont pas typées fortement.

#  Bash permet des opérations et des comparaisons d'entiers sur des variables
#+ contenant des caractères uniquements numériques.
#  Néanmoins, faites attention.

echo

if [ "$a" -ne "$b" ]
then
  echo "$a n'est pas égal à $b"
  echo "(comparaison arithmétique)"
fi

echo

if [ "$a" != "$b" ]
then
  echo "$a n'est pas égal à $b."
  echo "(comparaison de chaînes de caractères)"
  #     "4"  != "5"
  # ASCII 52 != ASCII 53
fi

# Pour cette instance particulière, "-ne" et "!=" fonctionnent.

echo

exit 0

Exemple 7.6. Vérification si une chaîne est nulle

#!/bin/bash
# str-test.sh: Tester des chaînes nulles et sans guillemets,
# "but not strings and sealing wax, not to mention cabbages and kings..."

# En utilisant   if [ ... ]


# Si une chaîne n'a pas été initialisée, elle n'a pas de valeur définie.
# Cet état est appelé "null" (ce qui n'est pas identique à zéro).

if [ -n $chaine1 ]    # $chaine1 n'est ni déclaré ni initialisé.
then
  echo "La chaîne \"chaine1\" n'est pas nulle."
else  
  echo "La chaîne \"chaine1\" est nulle."
fi  
# Mauvais résultat.
# Affiche $chaine1 comme non nulle bien qu'elle n'ait pas été initialisée.


echo


# Essayons de nouveau.

if [ -n "$chaine1" ]  # Cette fois, $chaine1 est entre guillemet.
then
  echo "La chaîne \"chaine1\" n'est pas nulle."
else  
  echo "La chaîne \"chaine1\" est nulle."
fi      # Entourer les chaînes avec des crochets de test.


echo


if [ $chaine1 ]       # Cette fois, $chaine1 est seule.
then
  echo "La chaîne \"chaine1\" n'est pas nulle."
else  
  echo "La chaîne \"chaine1\" est nulle."
fi  
# Ceci fonctionne.
# L'opérateur de test [ ] tout seul détecte si la chaîne est nulle.
# Néanmoins, une bonne pratique serait d'y mettre des guillemets ("$chaine1").
#
# Comme Stéphane Chazelas le dit,
#    if [ $chaine1 ]    a un argument, "]"
#    if [ "$chaine1" ]  a deux arguments, la chaîne "$chaine1" vide et "]" 



echo



chaine1=initialisée

if [ $chaine1 ]       # Une fois encore, $chaine1 est seule.
then
  echo "La chaîne \"chaine1\" n'est pas nulle."
else  
  echo "La chaîne \"chaine1\" est nulle."
fi  
# De nouveau, cela donne le résultat correct.
# Il est toujours préférable de la mettre entre guillemets ("$chaine1"), parce
# que...


chaine1="a = b"

if [ $chaine1 ]       # $chaine1 est de nouveau seule.
then
  echo "La chaîne \"chaine1\" n'est pas nulle."
else  
  echo "La chaîne \"chaine1\" est nulle."
fi  
# Ne pas mettre "$chaine1" entre guillemets donne un mauvais résultat !

exit 0
# Merci aussi à Florian Wisser pour le "heads up".

Exemple 7.7. zmore

#!/bin/bash

#Visualiser des fichiers gzip avec 'more'

SANSARGS=65
PASTROUVE=66
NONGZIP=67

if [ $# -eq 0 ] # même effet que:  if [ -z "$1" ]
# $1 peut exister mais doit être vide:  zmore "" arg2 arg3
then
  echo "Usage: `basename $0` nomfichier" >&2
  # Message d'erreur vers stderr.
  exit $SANSARGS
  # Renvoie 65 comme code de sortie du script (code d'erreur).
fi  

nomfichier=$1

if [ ! -f "$nomfichier" ]  #  Mettre $nomfichier entre guillemets permet d'avoir
                           #+ des espaces dans les noms de fichiers.
then
  echo "Fichier $nomfichier introuvable !" >&2
  # Message d'erreur vers stderr.
  exit $PASTROUVE
fi  

if [ ${nomfichier##*.} != "gz" ]
# Utilisation de crochets pour la substitution de variables.
then
  echo "Le fichier $1 n'est pas compressé avec gzip !"
  exit $NONGZIP
fi  

zcat $1 | more

# Utilise le filtre 'more'.
# Peut se substituer à 'less', si vous le souhaitez.


exit $?   # Le script renvoie le code d'erreur du tube.
#  En fait, "exit $?" n'est pas nécessaire, car le script retournera, pour
#+ chaque cas, le code de sortie de la dernière commande exécutée.

comparaison composée

-a

et logique

exp1 -a exp2 renvoie vrai si à la fois exp1 et exp2 sont vrais.

-o

ou logique

exp1 -o exp2 renvoie vrai si soit exp1 soit exp2 sont vrais.

Elles sont similaires aux opérateurs de comparaison Bash && et ||, utilisés à l'intérieur de double crochets.

[[ condition1 && condition2 ]]

Les opérateurs -o et -a fonctionnent avec la commande test ou à l'intérieur de simples crochets de test.

if [ "$exp1" -a "$exp2" ]

Référez-vous à l'Exemple 8.3, « Tests de conditions composées en utilisant && et || », à l'Exemple 26.15, « Simuler un tableau à deux dimensions, puis son test » et à l'Exemple A.30, « Chasse aux spammeurs » pour voir des opérateurs de comparaison composée en action.



[26] Comme S.C. l'a indiqué, dans un test composé, mettre la variable chaîne de caractères entre quotes pourrait ne pas suffire. [ -n "$chaine" -o "$a" = "$b" ] peut causer une erreur avec certaines versions de Bash si $chaine est vide. La façon la plus sûre est d'ajouter un caractère supplémentaire aux variables potentiellement vides, [ "x$chaine" != x -o "x$a" = "x$b" ] (les « x » sont annulés).