Exercices shell scripts
Entraînez-vous avec Bash (ou autres langages shell compatibles).
Cet article regroupe des exercices corrigés en shell scripts Bash et autres langages shell compatibles. Les exercices sont classés par niveaux et peuvent posséder plusieurs solutions.
Informations sur l'article
L'article a été originellement publié par Idriss Neumann sur developpez.com ici.
Niveau débutant
Exercice 1 - Appréciation de note
Énoncé
Créer un script qui demande à l'utilisateur de saisir une note et qui affiche un message en fonction de cette note :
- « très bien » si la note est entre 16 et 20
- « bien » lorsqu'elle est entre 14 et 16
- « assez bien » si la note est entre 12 et 14
- « moyen » si la note est entre 10 et 12
- « insuffisant » si la note est inférieur à 10
Solution
Solution proposée par Idriss Neumann :
#!/bin/bash
echo "Entrez votre note :"
read -r note
if [ "$note" -ge 16 ]; then
echo "très bien"
elif [ "$note" -ge 14 ]; then
echo "bien"
elif [ "$note" -ge 12 ]; then
echo "assez bien"
elif [ "$note" -ge 10 ]; then
echo "moyen"
else
echo "insuffisant"
fi
Exercice 2 - Appréciation de note (v2)
Énoncé
Reprenez l'exercice 1 et faites en sorte que le programme se répète tant que l'utilisateur n'a pas saisi une note négative ou q
(pour quitter).
Le script doit calculer le nombre de notes de saisies et en faire la moyenne tout à la fin.
Solution
Solution proposée par Idriss Neumann :
#!/bin/bash
note=0
moyenne=0
i=0
until [ "$note" -lt 0 ]; do
echo "Entrez votre note (q pour quitter) :"
read -r note
if [ "$note" = "q" ]; then
note=-1
echo "au revoir !"
elif [ "$note" -ge 16 ]; then
echo "très bien"
elif [ "$note" -ge 14 ]; then
echo "bien"
elif [ "$note" -ge 12 ]; then
echo "assez bien"
elif [ "$note" -ge 10 ]; then
echo "moyen"
elif [ "$note" -ge 0 ]; then
echo "insuffisant"
else
echo "au revoir !"
fi
if [ "$note" -ge 0 ]; then
let moyenne=$moyenne+$note
let i=$i+1
fi
done
if [ "$i" -le 0 ]; then
let i=1
fi
let moyenne=$moyenne/$i
echo "La moyenne est de $moyenne ($i notes)"
Exercice 3 - Nombre élevé à sa propre puissance
Énoncé
Créer un script qui prend un nombre en saisie et l'élève à sa propre puissance. C'est un peu le même principe que la factorielle mais cette fois, l'usage de la boucle for
est imposé.
Exemple d'exécution :
[ ~] ./NomDuScript.sh
Saisir une valeur :
2
2^2 = 4
Solution 1
Solution proposée par Idriss Neumann :
#!/bin/bash
echo "Saisir une valeur"
read -r value
result=1
for (( i=0 ; i<$value ; i++ )); do
let result=$result*$value
done
echo "$value^$value = $result"
Solution 2
Solution proposée par Idriss Neumann :
#!/bin/bash
operation () {
result=1
for (( i=0 ; i<$value ; i++ ))
do
let result=$result*$value
done
echo "$value^$value = $result"
}
if [ "$#" -eq 0 ]; then
echo "Saisir une valeur"
read -r value
else
value=$1
fi
operation
Niveau intermédiaire
Exercice 1 - Appréciation de note (v3)
Énoncé
Reprenez uniquement la version 1 de l'exercice. La note devra être donnée en paramètre ou bien saisie en cas d'absences d'arguments. La comparaison de la note devra être faite dans une fonction appreciation()
.
Solution
Solution proposée par Idriss Neumann :
#!/bin/bash
appreciation () {
if [ "$note" -ge 16 ]; then
echo "très bien"
elif [ "$note" -ge 14 ]; then
echo "bien"
elif [ "$note" -ge 12 ]; then
echo "assez bien"
elif [ "$note" -ge 10 ]; then
echo "moyen"
else
echo "insuffisant"
fi
}
# programme principal
clear
if [ "$#" -ne 0 ]; then
note=$1
else
echo "Saisir une note"
read -r note
fi
appreciation
Exercice 2 - TestUser
Énoncé
Créer un script qui vous propose le menu suivant :
1 - Vérifier l'existence d'un utilisateur
2 - Connaître l'UID d'un utilisateur
q - Quitter
L'utilisateur devra être saisi, à l'aide d'une fonction. Son existence devra être vérifiée à l'aide d'une autre fonction.
Solution
Solution proposée par Idriss Neumann :
#!/bin/bash
function pause {
echo "Appuyez sur ENTER pour continuer"
read
}
function saisirUser {
echo "Saisir l'utilisateur"
read -r util
}
function verifyUser {
if grep "^$util:" /etc/passwd > /dev/null; then
echo "L'utilisateur existe"
else
echo "L'utilisateur n'existe pas"
fi
pause
}
rep=1
while [ "$rep" -eq 1 ]; do
clear
printf "menu :\n\n"
echo "1. Vérifier l'existence d'un utilisateur"
echo "2. Connaître l'UID d'un utilisateur"
echo -e "3. Quitter\n"
read -r choix
case "$choix" in
1)
saisirUser
verifyUser ;;
2)
saisirUser
id $util
pause ;;
q)
echo "Au revoir"
pause
rep=0 ;;
*)
echo "Erreur de saisie"
pause ;;
esac
done
Exercice 3 - Calculatrice
Énoncé
Créer un script dans lequel deux nombres opérandes et un signe opérateur (+-*/) devront être donnés en paramètres, ou saisis. Le script doit réaliser l'opération souhaitée.
Exemple :
[ ~] ./calculette.sh 7 + 4
Le résultat est : 11
Le calcul devra être fait à l'aide d'une fonction calcul ()
.
Solution 1
Solution proposée par Idriss Neumann :
#!/bin/bash
saisir () {
printf "Saisir le premier nombre, puis le signe de l'opération puis le deuxième nombre :\n\n"
read -r nb1
read -r s
read -r nb2
}
calcul () {
case "$s" in
"+") let result=$nb1+$nb2 ;;
"-") let result=$nb1-$nb2 ;;
"*") let result=$nb1*$nb2 ;;
"/") let result=$nb1/$nb2 ;;
*)
let result=0
echo -e "Erreur de saisie !\nLe résultat est faux.";;
esac
}
calcul2 () {
let result=$nb1$s$nb2
}
if [ "$#" -eq 3 ]; then
nb1=$1 ; s=$2 ; nb2=$3
else
saisir
fi
calcul
echo "Le résultat est $result"
calcul2
echo "Calculé d'une autre façon : $result"
Solution 2
Solution proposée par Idriss Neumann :
#!/bin/bash
if [ "$#" -lt 3 ]; then
echo "Erreur : Il manque des paramètres !"
elif [[ "$1" =~ ^[0-9]+$ ]] && [[ "$3" =~ ^[0-9]+$ ]]; then
if [[ "$2" =~ ^(\+|\-|\/|\*){1}$ ]]; then
if [ $3 -ne 0 ] || [ "$2" != "/" ]; then
echo "Le résultat est : "$(( $1 $2 $3 ))
else
echo "Erreur : division par 0 !"
fi
else
echo "Erreur : opérateur invalide !"
fi
else
echo "Erreur : opérandes invalides !"
fi
Solution 3
Solution proposée par Idriss Neumann :
#!/bin/sh
if [ "$#" -lt 3 ]; then
echo "Erreur : Il manque des paramètres !"
elif echo "$1$3" | grep -E "^[0-9]{2,}$" > /dev/null; then
if echo "$2" | grep -E "^(\+|\-|\/|\*){1}$" > /dev/null; then
if [ $3 -ne 0 ] || [ "$2" != "/" ]; then
echo "Le résultat est : "$(( $1 $2 $3 ))
else
echo "Erreur : division par 0 !"
fi
else
echo "Erreur : opérateur invalide !"
fi
else
echo "Erreur : opérandes invalides !"
fi
Exercice 4 - La factorielle
Énoncé
Créer un script qui permet de calculer et d'afficher la factorielle d'un nombre donné en paramètre (ou saisi en cas d'absence de paramètres).
Solution 1
Solution proposée par Idriss Neumann :
#!/bin/bash
if [ "$#" -eq 0 ]; then
echo "Saisir une valeur : "
read -r val
else
val=$1
fi
# Dans le cas où c'est négatif, on rend la valeur positive
if [ "$val" -lt 0 ]; then
let val=-1*$val
fi
result=1
val2="$val"
while [ "$val" -ne 0 ]; do
printf "$val "
let result=$result*$val
let val=$val-1
if [ "$val" -ne 0 ]; then
printf "* "
fi
done
echo "= $result"
Solution 2
Solution proposée par Sve@r :
#!/bin/sh
if test "$#" -eq 0; then
echo "Saisissez une valeur correcte"
read -r val
set -- $val
fi
nb=${nb:-$1}
res=${res:-1}
if test "$nb" -eq 0; then
echo $res
exit
fi
res=`expr $res \* $nb`
nb=`expr $nb - 1`
. $0
Exercice 5 - Les fichiers
Énoncé
Créer un script qui doit calculer le nombre de fichiers standard, de sous-répertoires, et d'exécutables d'un répertoire quelconque qui sera donné en paramètre (ou saisis en cas d'absence du paramètre).
Solution
Solution proposée par Idriss Neumann :
#!/bin/bash
j=0
k=0
l=0
if [ "$#" -eq 0 ]; then
echo "Saisir le répertoire"
read -r rep
else
rep=$1
fi
cd $rep
for i in *; do
if [ -d "$i" ]; then
echo "$i"
let j=$j+1
fi
if [ -f "$i" ]; then
echo $i
let k=$k+1
fi
if [ -x "$i" ]; then
echo $i
let l=$l+1
fi
done
echo "Il y a $j répertoires, $k fichiers et $l exécutables dans $rep"
Exercice 6 - Tri à bulle
Énoncé
Créer un script qui devra enregistrer à l'aide d'un tableau, un nombre d'entiers donné en paramètre (ou en saisie) puis trier ceux-ci dans l'ordre croissant dans ce même tableau (sans passer par un autre) et enfin afficher le contenu du tableau (ordonné) sur la sortie standard.
Solution
Solution proposée par Idriss Neumann :
#!/bin/bash
if [ "$#" -lt 1 ]; then
echo "Saisir le nombre d'éléments à ordonner"
read -r SIZE
else
SIZE=$1
fi
echo "Saisir les éléments :"
for (( i=0 ; i<SIZE ; i++ )); do
read -r tab[i]
done
for (( i=0 ; i<SIZE ; i++ )); do
for (( j=$i+1 ; j<SIZE ; j++ ))
do
if [ "${tab[$j]}" -le "${tab[$i]}" ]; then
tampon="${tab[$i]}"
tab[$i]="${tab[$j]}"
tab[$j]="$tampon"
fi
done
done
echo "Valeurs ordonnées :"
for (( i=0 ; i<SIZE ; i++ )); do
echo "${tab[$i]}"
done
Exercice 7 - ls avec ordre inversé
Énoncé
Créer un script qui renvoie la même sortie que la commande ls
, mais dans l'ordre décroissant (autrement dit : le script devra lister le contenu d'un répertoire dans l'ordre décroissant). Vous ne devez ni vous servir de la commande ls
, ni de la commande sort
.
Solution 1
Solution proposée par Idriss Neumann :
#!/bin/bash
# Si on a au moins un paramètre et que le premier paramètre est un répertoire
if [ "$#" -ge 1 ] && [ -d "$1v ]; then
cd $1
fi
nb=0
for fichier in *; do
tab[$nb]="$fichier"
let nb=$nb+1
done
# affichage inversé
for (( i=$nb ; i>=0 ; i=$i-1 )); do
echo ${tab[$i]}
done
Solution 2
Solution proposée par becket :
#!/bin/bash
listing=( * )
for (( i=${#listing[@]}-1 ; i >= 0 ; i=i-1 )); do
echo ${listing[$i]}
done
Solution 3
Solution proposée par N_Bah :
#!/bin/bash
read -rp 'Entrez le nom du répertoire : ' repertoire
if [ -d "$repertoire" ]; then
[[ $repertoire != */ ]] && repertoire="${repertoire:+$repertoire/}"
shopt -s nullglob #sinon repertoireVide/*, retournera repertoireVide/*
fichiers=( ${repertoire}* )
((${#fichiers[@]})) && {
for (( x=${#fichiers[@]}-1; x>=0; x-- )); do
echo "${fichiers[x]}"
done
} || echo "$repertoire est vide"
else
echo "$repertoire n'est pas un répertoire"
fi
Solution 4
Solution proposée par chardclo :
#!/bin/bash
read -rp 'Entrez le nom du répertoire : ' repertoire
[ -d "$repertoire" ] || {
printf "%s n'est pas un nom de dossier valide." "$repertoire"
exit 1
}
compgen -o default "${repertoire}/" | tac
Solution 5
Solution proposée par chardclo :
#!/bin/bash
read -rp 'Entrez le nom du répertoire : ' repertoire
[ -d "$repertoire" ] || {
printf "%s n'est pas un nom de dossier valide.\n" "$repertoire"
exit 1
}
text=""
while read -r; do
text="${REPLY}\n${text}"
done < <(compgen -o default "${repertoire}/")
[[ ${text} ]] && printf "$text" || printf "le dossier %s est vide.\n" "$repertoire"
Niveau confirmé
Exercice 1 - Fichier de notes
Énoncé
Créer un script qui va devra exploiter les données d'un fichier de notes que vous allez créer au préalable dans le même répertoire que le script.
Ce fichier sera appelé FichierNote.txt et devra se présenter comme ceci :
Dupont François 12
Durand Françoise 8
Dujardin Nicole 14
Le script devra afficher les lignes dans lesquelles la note est supérieure ou égale à 10.
Exemple :
[ ~] ./NomDuScript
Dupont François 12
Dujardin Nicole 14
Solution
Solution proposée par Idriss Neumann :
#!/bin/bash
fichier="FichierNote.txt"
while read -r ligne; do
set -- "$ligne"
if [ "$3" -ge 10 ]; then
echo "$ligne"
fi
done < $fichier
Exercice 2 - Liste d'utilisateurs
Énoncé
Créer un script qui prend en paramètre (ou en saisie en cas d'absence du paramètre) un fichier qui contient des lignes comme ceci : Login + Tabulation + Nom + Tabulation + Prénom.
Exemple :
dupontf Dupont François
fdurand Durand Françoise
nicoled Dujardin Nicole
Le script devra vérifier, à l'aide d'une fonction, l'existence des utilisateurs enregistrés dans le fichier. Admettons, par exemple, que seul Dupont François soit un utilisateur et que le fichier se nomme ~/Documents/FichierUser
, le script devra s'exécuter comme ceci :
[ ~] ./NomDuScript ~/Doucuments/FichierUser
dupontf Dupont François
[ ~] ./NomDuScript
Saisissez le fichier à traiter :
~/Documents/UserFichier
Le fichier n'existe pas.
[ ~]
Le script devra donc, au préalable, vérifier l'existence du fichier avant de comparer son contenu au fichier /etc/passwd
. Le script devra également quitter la boucle si le fichier est vide.
Solution
Solution proposée par Idriss Neumann :
#!/bin/bash
TestUser () {
if grep "^$util:" /etc/passwd > /dev/null
then
echo $ligne
fi
}
if [ "$#" -eq 0 ]; then
echo "Chemin et nom du fichier :"
read -r fichier
else
fichier="$1"
fi
if [ -e "$fichier" ]; then
while read -r ligne; do
set -- "$ligne"
util="$1"
TestUser
done < $fichier
else
echo "Le fichier $fichier n'existe pas."
fi
Exercice 3 - Convertisseur décimal/binaire
Énoncé
Créer un script qui prend en paramètre (ou en saisie en cas d'absence de paramètres) une valeur décimale et qui doit la convertir en binaire.
Vous devez travailler sur 8 bits et chaque bit devra être contenu dans une case d'un tableau monodimensionnel et à la fin on affiche toutes les cases de ce tableau pour avoir la valeur en binaire lisible de droite à gauche, à partir de la valeur décimale saisie au départ.
Solution 1
Solution propos ée par Idriss Neumann :
#!/bin/bash
MaxBits=8
pow () {
value2=1
for (( k=1 ; k<$i ; k ++ )); do
let value2=$value2*2
done
}
if [ "$#" -eq 0 ]; then
echo "Saisir une valeur décimale"
read -r value
else
value="$1"
fi
declare -a bin
j=0
for (( i="$MaxBits" ; i>=0 ; i-- )); do
pow
if [ "$value" -ge "$value2" ]; then
bin[$j]="1"
let value=$value-$value2
else
bin[$j]="0"
fi
let j=$j+1
done
printf "La valeur binaire est de : "
for (( i=0 ; i<$MaxBits ; i++ )); do
printf "${bin[$i]}"
done
echo ""
Solution 2
Solution proposée par becket :
#!/bin/bash
unset resultat
echo "Entrez une valeur a convertir : "
read -r i
for (( cpt=8 ; $cpt > 1 ; cpt-- )); do
let resultat[$cpt]="$i & 1"
let i="i >>= 1"
done
echo "${resultat[@]}"
Solution 3
Solution proposée par chardclo :
#!/bin/bash
unset res
dec=$1
[[ $dec ]] || read -rp "Entrez une valeur a convertir : " dec
while true; do
res[$dec]=$((dec & 1))
((dec>>=1 )) || break
done
echo "${res[@]}"
Exercice 4 - Moyenne de notes sur un fichier
Énoncé
Créer un script qui prend en paramètre ou en saisie le nom d'un fichier contenant le nom des élèves et leurs trois notes. Le script devra : afficher les noms des élèves, puis calculer et afficher la moyenne de chaque élève.
Voici comment se présente le fichier :
Durand 12 9 14
Lucas 8 11 4
Martin 9 12 1
Solution 1
Solution proposée par Idriss Neumann :
#!/bin/bash
if [ "$#" -lt 1 ]; then
echo "Saisir le nom du fichier"
read -r fichier
else
fichier="$1"
fi
while read -r ligne; do
set -- "$ligne"
let moyenne=($2+$3+$4)/3
echo "L'élève $1 a pour moyenne $moyenne"
done < "$fichier"
Solution 2
Solution proposée par N_Bah :
#!/bin/bash
moyenne() {
declare -i somme
for i in "$@"; do
somme+="$i"
done
echo "$((somme / ${#@}))"
}
[ -f "$1" ] && fichier="$1" || read -rep 'Entrez le nom du fichier qui contient les données : ' fichier
while read -r nom notes; do
echo -n "$nom : "
moyenne $notes
done < "$fichier"
Liens utiles
Quelques liens utiles permettant d'acquérir de bonnes bases :
- Advanced Bash-Scripting Guide (traduction)
- Un cours complet sur la programmation Shell
- Quelques bonnes pratiques dans l'écriture de scripts en Bash
- Présentation et cours Korn Shell (compatible avec le Bash)
- La section « Le Shell » de la FAQ Linux
Remerciements
Je souhaite remercier tous les contributeurs qui m'ont aidé à enrichir cet article par des exercices et/ou des solutions. Il s'agit en l'occurrence de becket, chardclo, N_Bah et Sve@r merci à eux.
Je tiens également à remercier LittleWhite, Louson et paissad pour leur relecture technique et leurs conseils.
Je tiens enfin à remercier jacques_jean pour son effort de relecture orthographique.