5.2. Échappement

Échapper est une méthode pour mettre entre guillemets un caractère seul. L'échappement (\) précédant un caractère dit au shell d'interpréter le caractère littéralement.

[Attention]

Attention

Avec certaines commandes et utilitaires, tels que echo et sed, échapper un caractère peut avoir l'effet inverse - cela peut activer un comportement particulier pour ce caractère.

Significations spéciales des caractères échappés

utilisé avec echo et sed
\n

passe à la ligne

\r

renvoie le curseur en début de ligne

\t

tabulation

\v

tabulation verticale

\b

retour en arrière

\a

« alerte » (sonore ou visuelle)

\0xx

Transcode en octal le caractère dont le code ASCII est 0xx

Exemple 5.2. Caractères d'échappement

#!/bin/bash
# escaped.sh: caractères d'échappement

echo; echo

# Échapper un retour à la ligne.
# ------------------------------

echo ""

echo "Ceci s'affichera
sur deux lignes."
# Ceci s'affichera
# sur deux lignes.

echo "Ceci s'affichera \
sur une lignes."
# Ceci s'affichera sur une ligne.

echo; echo

echo "============="


echo "\v\v\v\v"      # Affiche \v\v\v\v litérallement.
# Utilisez l'option -e avec 'echo' pour afficher les caractères d'échappement.
echo "======================"
echo "TABULATIONS VERTICALES"
echo -e "\v\v\v\v"   # Affiche 4 tabulations verticales.
echo "======================="

echo "GUILLEMET DOUBLE"
echo -e "\042"       # Affiche " (guillemet, caractère octal ASCII 42).
echo "======================="

# La construction $'\X' rend l'option -e inutile.
echo; echo "RETOUR CHARIOT ET SON"
echo $'\n'           # Retour chariot.
echo $'\a'           # Alerte (son).

echo "======================="
echo "GUILLEMETS"
#  Les version 2 et ultérieures de Bash permettent l'utilisation de la
#+ construction $'\nnn'.
# Notez que, dans ce cas, '\nnn' est une valeur octale.
echo $'\t \042 \t'   # Guillemet (") entouré par des tabulations.

#  Cela fonctionne aussi avec des valeurs hexadécimales, dans une construction
#+ du type $'\xhhh'.
echo $'\t \x22 \t'  # Guillemet (") entouré par des tabulations.
# Merci, Greg Keraunen, pour nous l'avoir indiqué.
# Les versions précédentes de Bash permettent '\x022'.
echo "======================="
echo


# Affecter des caractères ASCII à une variable.
# ----------------------------------------------
guillemet=$'\042'        # " affecté à une variable.
echo "$guillemet Ceci est un chaîne de caractères mise entre guillemets, $guillemet " \
  "et ceci reste en dehors des guillemets."

echo

# Concaténer des caractères ASCII dans une variable.
trois_soulignes=$'\137\137\137'  # 137 est le code octal ASCII pour '_'.
echo "$trois_soulignes SOULIGNE $trois_soulignes"

echo

ABC=$'\101\102\103\010'  # 101, 102, 103 sont les codes octals de A, B, C.
echo $ABC

echo; echo

echappe=$'\033'                    # 033 est le code octal pour l'échappement.
echo "\"echappe\" s'affiche comme $echappe"
#                                   pas de sortie visible.

echo; echo

exit 0

Voir l'Exemple 34.1, « Expansion de chaîne de caractères » pour un autre exemple d'extension de chaînes avec la structure $' '.

\"

donne au guillemet sa signification littérale

echo "Bonjour"                    # Bonjour
echo "\"Bonjour\", a-t-il dit."  # "Bonjour", a-t-il dit.
\$

donne au dollar sa signification littérale (un nom de variable suivant un \$ ne sera pas référencé)

echo "\$variable01"  # donne $variable01
\\

donne à l'antislash sa signification littérale

echo "\\"  # donne \

# Alors que...

echo "\"   # Appelle une deuxième invite de la ligne de commande.
           # Dans un script, donne un message d'erreur.
[Note]

Note

Le comportement de \ est dicté par son « auto-échappement », sa mise entre guillemets ou son apparition dans une substitution de commandes ou dans un document en ligne.

                      #  Simple échappement et mise entre guillemets
echo \z               #  z
echo \\z              # \z
echo '\z'             # \z
echo '\\z'            # \\z
echo "\z"             # \z
echo "\\z"            # \z

                      #  Substitution de commandes
echo `echo \z`        #  z
echo `echo \\z`       #  z
echo `echo \\\z`      # \z
echo `echo \\\\z`     # \z
echo `echo \\\\\\z`   # \z
echo `echo \\\\\\\z`  # \\z
echo `echo "\z"`      # \z
echo `echo "\\z"`     # \z

                      # Document en ligne
cat <<EOF
\z
EOF                   # \z

cat <<EOF
\\z
EOF                   # \z

# Ces exemples ont été donnés par Stéphane Chazelas.

Les éléments d'une chaîne de caractères affectée à une variable peuvent être échappés, mais le caractère d'échappement seul ne devrait pas être affecté à une variable.

variable=\
echo "$variable"
# Ne fonctionne pas et donne le message d'erreur :
# test.sh: : command not found
# Un échappement seul ne peut être affecté correctement à une variable.
#
#  Ce qui arrive ici est que "\" échappe le saut de ligne et
#+ l'effet est          variable=echo "$variable"
#+                      affectation invalide de variable

variable=\
23skidoo
echo "$variable"        #  23skidoo
                        #  Ça fonctionne car la deuxième ligne est une affectation
                        #+ valide de variable.

variable=\
#        \^    échappement suivi d'une espace
echo "$variable"        # espace

variable=\\
echo "$variable"        # \

variable=\\\
echo "$variable"
# Ne fonctionnera pas et donne le message d'erreur :
# test.sh: \: command not found
#
#  La première séquence d'échappement échappe la deuxième, mais la troisième est laissée
#+ seule avec le même résultat que dans le premier exemple ci-dessus.

variable=\\\\
echo "$variable"        # \\
                        # Deuxième et quatrième séquences d'échappement
                        # Ça marche.

Échapper une espace peut empêcher la séparation de mots dans une liste d'arguments pour une commande.

liste_fichiers="/bin/cat /bin/gzip /bin/more /usr/bin/less /usr/bin/emacs-20.7"
# Liste de fichiers comme argument(s) d'une commande.

# On demande de tout lister, avec deux fichiers en plus.
ls -l /usr/X11R6/bin/xsetroot /sbin/dump $file_list

echo "-------------------------------------------------------------------------"

# Qu'arrive-t'il si nous échappons un ensemble d'espaces ?
ls -l /usr/X11R6/bin/xsetroot\ /sbin/dump\ $file_list
# Erreur: les trois premiers fichiers sont concaténés en un seul argument pour 'ls -l'
#        parce que les deux espaces échappés empêchent la séparation des arguments (mots).

L'échappement permet également d'écrire une commande sur plusieurs lignes. Normalement, chaque ligne séparée constitue une commande différente mais un échappement à la fin d'une ligne échappe le caractère de saut de ligne, et la séquence de la commande continue sur la ligne suivante.

(cd /source/repertoire && tar cf - . ) | \
(cd /dest/repertoire && tar xpvf -)
# Répétant la commande de copie de répertoires d'Alan Cox, mais séparée en deux lignes
# pour accroître la lisibilité.

# Comme alternative :
tar cf - -C /source/directory . |
tar xpvf - -C /dest/directory
# Voir note ci-dessous.
# (Merci, Stéphane Chazelas.)
[Note]

Note

Si la ligne d'un script termine avec un |, le caractère tube, alors il n'est pas strictement nécessaire de mettre un échappement \. Il est néanmoins considéré comme une bonne pratique de programmation de toujours échapper une ligne de code qui continue sur la ligne suivante.

echo "foo
bar"
#foo
#bar

echo

echo 'foo
bar'    # Pas encore de différence.
#foo
#bar

echo

echo foo\
bar     # Saut de ligne échappé.
#foobar

echo

echo "foo\
bar"     #  Pareil ici, car \ toujours interpreté comme un échappement à l'intérieur de
         #  guillemets faibles.
#foobar

echo

echo 'foo\
bar'     # Le caractère d'échappement \ est pris littéralement à cause des guillemets forts.
#foo\
#bar

# Exemples suggérés par Stéphane Chazelas.