Patrice Ferlet
Patrice Ferlet
Créateur de ce blog.
Publié le mars 25, 2009 Temps de lecture: 2 min

Me suivre sur Mastodon

Traiter des listes correctement en Bash

Bien souvent j’utilise des fonctions de boucle en Bash pour traiter des fichiers ou faire des tests. Vous pouvez utiliser une liste que vous donnez manuellement, une liste de fichier ou carrément utiliser une commande qui génère la liste.

Restons dans la simplicité, voici un ou deux exemples: ``

for i in “foo bar baz”

do echo $i done

affiche:

foo

bar

baz

#for i in find ./ -name "*.php" do cp $i ~/sauvegarde_php sed -i “s/\r//g” $i
done

copie chaque fichier php dans un répertoire de sauvegarde puis remplace “\r” dans chaque fichier trouvé

#for i in cat monfichier.txt do grep $i autrefichier.txt || echo “$i not found !” done

à chaque ligne trouvé dans monfichier.txt je vérifie si elle existe aussi dans autrefichier.txt sinon j’affiche le souci ``

Là où on peut avoir un souci, c’est justement dans le dernier exemple. Imaginons que monfichier.txt contienne:

foo
bar
baz

et le script: #for i in `cat monfichier.txt` do echo $i done Dans ce cas, on retrouve bien une ligne pour chaque élément.

Mais un problème intervient dans ce cas:

une ligne à chercher
une autre ligne
la dernière ligne

Le résultat va être:

une
ligne
à
chercher
une
autre
ligne
la
dernière
ligne

Stupeur… tremblements… ce n’est pas une erreur mais le fonctionnement normal du principe de liste en Bash. En effet, la séparation des éléments se fait via des retours chariots **ou** des espaces…

Voici comment vous pouvez vous en sortir: utiliser la commande “read” qui attend bien un retour chariot (ou \n) pour passer à la suite: cat monfichier.txt | while read i do echo $i done Et on retrouve notre liste sans coupure à chaque espace mais bien pour chaque ligne.

Efficace non ?

comments powered by Disqus